Coverage Report

Created: 2025-08-28 06:57

/src/gdal/gcore/gdalnodatamaskband.cpp
Line
Count
Source (jump to first uncovered line)
1
/******************************************************************************
2
 *
3
 * Project:  GDAL Core
4
 * Purpose:  Implementation of GDALNoDataMaskBand, a class implementing all
5
 *           a default band mask based on nodata values.
6
 * Author:   Frank Warmerdam, warmerdam@pobox.com
7
 *
8
 ******************************************************************************
9
 * Copyright (c) 2007, Frank Warmerdam
10
 * Copyright (c) 2008-2012, Even Rouault <even dot rouault at spatialys.com>
11
 *
12
 * SPDX-License-Identifier: MIT
13
 ****************************************************************************/
14
15
#include "cpl_port.h"
16
#include "gdal_priv.h"
17
18
#include <algorithm>
19
#include <cassert>
20
#include <cmath>
21
#include <cstring>
22
#include <utility>
23
24
#include "cpl_conv.h"
25
#include "cpl_error.h"
26
#include "cpl_vsi.h"
27
#include "gdal.h"
28
#include "gdal_priv_templates.hpp"
29
30
//! @cond Doxygen_Suppress
31
/************************************************************************/
32
/*                        GDALNoDataMaskBand()                          */
33
/************************************************************************/
34
35
GDALNoDataMaskBand::GDALNoDataMaskBand(GDALRasterBand *poParentIn)
36
0
    : m_poParent(poParentIn)
37
0
{
38
0
    poDS = nullptr;
39
0
    nBand = 0;
40
41
0
    nRasterXSize = m_poParent->GetXSize();
42
0
    nRasterYSize = m_poParent->GetYSize();
43
44
0
    eDataType = GDT_Byte;
45
0
    m_poParent->GetBlockSize(&nBlockXSize, &nBlockYSize);
46
47
0
    const auto eParentDT = m_poParent->GetRasterDataType();
48
0
    if (eParentDT == GDT_Int64)
49
0
        m_nNoDataValueInt64 = m_poParent->GetNoDataValueAsInt64();
50
0
    else if (eParentDT == GDT_UInt64)
51
0
        m_nNoDataValueUInt64 = m_poParent->GetNoDataValueAsUInt64();
52
0
    else
53
0
        m_dfNoDataValue = m_poParent->GetNoDataValue();
54
0
}
55
56
/************************************************************************/
57
/*                        GDALNoDataMaskBand()                          */
58
/************************************************************************/
59
60
GDALNoDataMaskBand::GDALNoDataMaskBand(GDALRasterBand *poParentIn,
61
                                       double dfNoDataValue)
62
0
    : m_poParent(poParentIn)
63
0
{
64
0
    poDS = nullptr;
65
0
    nBand = 0;
66
67
0
    nRasterXSize = m_poParent->GetXSize();
68
0
    nRasterYSize = m_poParent->GetYSize();
69
70
0
    eDataType = GDT_Byte;
71
0
    m_poParent->GetBlockSize(&nBlockXSize, &nBlockYSize);
72
73
0
    const auto eParentDT = m_poParent->GetRasterDataType();
74
0
    if (eParentDT == GDT_Int64)
75
0
        m_nNoDataValueInt64 = static_cast<int64_t>(dfNoDataValue);
76
0
    else if (eParentDT == GDT_UInt64)
77
0
        m_nNoDataValueUInt64 = static_cast<uint64_t>(dfNoDataValue);
78
0
    else
79
0
        m_dfNoDataValue = dfNoDataValue;
80
0
}
81
82
/************************************************************************/
83
/*                       ~GDALNoDataMaskBand()                          */
84
/************************************************************************/
85
86
0
GDALNoDataMaskBand::~GDALNoDataMaskBand() = default;
87
88
/************************************************************************/
89
/*                          GetWorkDataType()                           */
90
/************************************************************************/
91
92
static GDALDataType GetWorkDataType(GDALDataType eDataType)
93
0
{
94
0
    GDALDataType eWrkDT = GDT_Unknown;
95
0
    switch (eDataType)
96
0
    {
97
0
        case GDT_Byte:
98
0
            eWrkDT = GDT_Byte;
99
0
            break;
100
101
0
        case GDT_Int16:
102
0
            eWrkDT = GDT_Int16;
103
0
            break;
104
105
0
        case GDT_UInt16:
106
0
            eWrkDT = GDT_UInt16;
107
0
            break;
108
109
0
        case GDT_UInt32:
110
0
            eWrkDT = GDT_UInt32;
111
0
            break;
112
113
0
        case GDT_Int8:
114
0
        case GDT_Int32:
115
0
        case GDT_CInt16:
116
0
        case GDT_CInt32:
117
0
            eWrkDT = GDT_Int32;
118
0
            break;
119
120
0
        case GDT_Float16:
121
0
        case GDT_CFloat16:
122
0
        case GDT_Float32:
123
0
        case GDT_CFloat32:
124
0
            eWrkDT = GDT_Float32;
125
0
            break;
126
127
0
        case GDT_Float64:
128
0
        case GDT_CFloat64:
129
0
            eWrkDT = GDT_Float64;
130
0
            break;
131
132
0
        case GDT_Int64:
133
0
        case GDT_UInt64:
134
0
            eWrkDT = eDataType;
135
0
            break;
136
137
0
        case GDT_Unknown:
138
0
        case GDT_TypeCount:
139
0
            CPLAssert(false);
140
0
            eWrkDT = GDT_Float64;
141
0
            break;
142
0
    }
143
0
    return eWrkDT;
144
0
}
145
146
/************************************************************************/
147
/*                          IsNoDataInRange()                           */
148
/************************************************************************/
149
150
bool GDALNoDataMaskBand::IsNoDataInRange(double dfNoDataValue,
151
                                         GDALDataType eDataTypeIn)
152
0
{
153
0
    GDALDataType eWrkDT = GetWorkDataType(eDataTypeIn);
154
0
    switch (eWrkDT)
155
0
    {
156
0
        case GDT_Byte:
157
0
        {
158
0
            return GDALIsValueInRange<GByte>(dfNoDataValue);
159
0
        }
160
161
0
        case GDT_Int8:
162
0
        {
163
0
            return GDALIsValueInRange<signed char>(dfNoDataValue);
164
0
        }
165
166
0
        case GDT_Int16:
167
0
        {
168
0
            return GDALIsValueInRange<GInt16>(dfNoDataValue);
169
0
        }
170
171
0
        case GDT_UInt16:
172
0
        {
173
0
            return GDALIsValueInRange<GUInt16>(dfNoDataValue);
174
0
        }
175
176
0
        case GDT_UInt32:
177
0
        {
178
0
            return GDALIsValueInRange<GUInt32>(dfNoDataValue);
179
0
        }
180
0
        case GDT_Int32:
181
0
        {
182
0
            return GDALIsValueInRange<GInt32>(dfNoDataValue);
183
0
        }
184
185
0
        case GDT_UInt64:
186
0
        {
187
0
            return GDALIsValueInRange<uint64_t>(dfNoDataValue);
188
0
        }
189
190
0
        case GDT_Int64:
191
0
        {
192
0
            return GDALIsValueInRange<int64_t>(dfNoDataValue);
193
0
        }
194
195
0
        case GDT_Float16:
196
0
        {
197
0
            return std::isnan(dfNoDataValue) || std::isinf(dfNoDataValue) ||
198
0
                   GDALIsValueInRange<GFloat16>(dfNoDataValue);
199
0
        }
200
201
0
        case GDT_Float32:
202
0
        {
203
0
            return std::isnan(dfNoDataValue) || std::isinf(dfNoDataValue) ||
204
0
                   GDALIsValueInRange<float>(dfNoDataValue);
205
0
        }
206
207
0
        case GDT_Float64:
208
0
        {
209
0
            return true;
210
0
        }
211
212
0
        case GDT_CFloat16:
213
0
        case GDT_CFloat32:
214
0
        case GDT_CFloat64:
215
0
        case GDT_CInt16:
216
0
        case GDT_CInt32:
217
0
        case GDT_Unknown:
218
0
        case GDT_TypeCount:
219
0
            break;
220
0
    }
221
222
0
    CPLAssert(false);
223
0
    return false;
224
0
}
225
226
/************************************************************************/
227
/*                             IReadBlock()                             */
228
/************************************************************************/
229
230
CPLErr GDALNoDataMaskBand::IReadBlock(int nXBlockOff, int nYBlockOff,
231
                                      void *pImage)
232
233
0
{
234
0
    const int nXOff = nXBlockOff * nBlockXSize;
235
0
    const int nXSizeRequest = std::min(nBlockXSize, nRasterXSize - nXOff);
236
0
    const int nYOff = nYBlockOff * nBlockYSize;
237
0
    const int nYSizeRequest = std::min(nBlockYSize, nRasterYSize - nYOff);
238
239
0
    if (nBlockXSize != nXSizeRequest || nBlockYSize != nYSizeRequest)
240
0
    {
241
0
        memset(pImage, 0, static_cast<GPtrDiff_t>(nBlockXSize) * nBlockYSize);
242
0
    }
243
244
0
    GDALRasterIOExtraArg sExtraArg;
245
0
    INIT_RASTERIO_EXTRA_ARG(sExtraArg);
246
0
    return IRasterIO(GF_Read, nXOff, nYOff, nXSizeRequest, nYSizeRequest,
247
0
                     pImage, nXSizeRequest, nYSizeRequest, GDT_Byte, 1,
248
0
                     nBlockXSize, &sExtraArg);
249
0
}
250
251
/************************************************************************/
252
/*                            SetZeroOr255()                            */
253
/************************************************************************/
254
255
#if (defined(__GNUC__) && !defined(__clang__))
256
__attribute__((optimize("tree-vectorize")))
257
#endif
258
static void
259
SetZeroOr255(GByte *pabyDestAndSrc, size_t nBufSize, GByte byNoData)
260
0
{
261
0
    for (size_t i = 0; i < nBufSize; ++i)
262
0
    {
263
0
        pabyDestAndSrc[i] = (pabyDestAndSrc[i] == byNoData) ? 0 : 255;
264
0
    }
265
0
}
266
267
template <class T>
268
#if (defined(__GNUC__) && !defined(__clang__))
269
__attribute__((optimize("tree-vectorize")))
270
#endif
271
static void
272
SetZeroOr255(GByte *pabyDest, const T *panSrc, size_t nBufSize, T nNoData)
273
0
{
274
0
    for (size_t i = 0; i < nBufSize; ++i)
275
0
    {
276
0
        pabyDest[i] = (panSrc[i] == nNoData) ? 0 : 255;
277
0
    }
278
0
}
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned char>(unsigned char*, unsigned char const*, unsigned long, unsigned char)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<short>(unsigned char*, short const*, unsigned long, short)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned short>(unsigned char*, unsigned short const*, unsigned long, unsigned short)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned int>(unsigned char*, unsigned int const*, unsigned long, unsigned int)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<int>(unsigned char*, int const*, unsigned long, int)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<long>(unsigned char*, long const*, unsigned long, long)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned long>(unsigned char*, unsigned long const*, unsigned long, unsigned long)
279
280
template <class T>
281
static void SetZeroOr255(GByte *pabyDest, const T *panSrc, int nBufXSize,
282
                         int nBufYSize, GSpacing nPixelSpace,
283
                         GSpacing nLineSpace, T nNoData)
284
0
{
285
0
    if (nPixelSpace == 1)
286
0
    {
287
0
        for (int iY = 0; iY < nBufYSize; iY++)
288
0
        {
289
0
            SetZeroOr255(pabyDest, panSrc, nBufXSize, nNoData);
290
0
            pabyDest += nLineSpace;
291
0
            panSrc += nBufXSize;
292
0
        }
293
0
    }
294
0
    else
295
0
    {
296
0
        size_t i = 0;
297
0
        for (int iY = 0; iY < nBufYSize; iY++)
298
0
        {
299
0
            GByte *pabyLineDest = pabyDest + iY * nLineSpace;
300
0
            for (int iX = 0; iX < nBufXSize; iX++)
301
0
            {
302
0
                *pabyLineDest = (panSrc[i] == nNoData) ? 0 : 255;
303
0
                ++i;
304
0
                pabyLineDest += nPixelSpace;
305
0
            }
306
0
        }
307
0
    }
308
0
}
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned char>(unsigned char*, unsigned char const*, int, int, long long, long long, unsigned char)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<short>(unsigned char*, short const*, int, int, long long, long long, short)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned short>(unsigned char*, unsigned short const*, int, int, long long, long long, unsigned short)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned int>(unsigned char*, unsigned int const*, int, int, long long, long long, unsigned int)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<int>(unsigned char*, int const*, int, int, long long, long long, int)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<long>(unsigned char*, long const*, int, int, long long, long long, long)
Unexecuted instantiation: gdalnodatamaskband.cpp:void SetZeroOr255<unsigned long>(unsigned char*, unsigned long const*, int, int, long long, long long, unsigned long)
309
310
/************************************************************************/
311
/*                             IRasterIO()                              */
312
/************************************************************************/
313
314
CPLErr GDALNoDataMaskBand::IRasterIO(GDALRWFlag eRWFlag, int nXOff, int nYOff,
315
                                     int nXSize, int nYSize, void *pData,
316
                                     int nBufXSize, int nBufYSize,
317
                                     GDALDataType eBufType,
318
                                     GSpacing nPixelSpace, GSpacing nLineSpace,
319
                                     GDALRasterIOExtraArg *psExtraArg)
320
0
{
321
0
    if (eRWFlag != GF_Read)
322
0
    {
323
0
        return CE_Failure;
324
0
    }
325
0
    const auto eParentDT = m_poParent->GetRasterDataType();
326
0
    const GDALDataType eWrkDT = GetWorkDataType(eParentDT);
327
328
    // Optimization in common use case (#4488).
329
    // This avoids triggering the block cache on this band, which helps
330
    // reducing the global block cache consumption.
331
0
    if (eBufType == GDT_Byte && eWrkDT == GDT_Byte && nPixelSpace == 1 &&
332
0
        nLineSpace >= nBufXSize)
333
0
    {
334
0
        const CPLErr eErr = m_poParent->RasterIO(
335
0
            GF_Read, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize, nBufYSize,
336
0
            eBufType, nPixelSpace, nLineSpace, psExtraArg);
337
0
        if (eErr != CE_None)
338
0
            return eErr;
339
340
0
        GByte *pabyData = static_cast<GByte *>(pData);
341
0
        const GByte byNoData = static_cast<GByte>(m_dfNoDataValue);
342
343
0
        if (nLineSpace == nBufXSize)
344
0
        {
345
0
            const size_t nBufSize = static_cast<size_t>(nBufXSize) * nBufYSize;
346
0
            SetZeroOr255(pabyData, nBufSize, byNoData);
347
0
        }
348
0
        else
349
0
        {
350
0
            assert(nLineSpace > nBufXSize);
351
0
            for (int iY = 0; iY < nBufYSize; iY++)
352
0
            {
353
0
                SetZeroOr255(pabyData, nBufXSize, byNoData);
354
0
                pabyData += nLineSpace;
355
0
            }
356
0
        }
357
0
        return CE_None;
358
0
    }
359
360
0
    const auto AllocTempBufferOrFallback =
361
0
        [this, eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize,
362
0
         nBufYSize, eBufType, nPixelSpace, nLineSpace,
363
0
         psExtraArg](int nWrkDTSize) -> std::pair<CPLErr, void *>
364
0
    {
365
0
        auto poParentDS = m_poParent->GetDataset();
366
        // Check if we must simulate a memory allocation failure
367
        // Before checking the env variable, which is slightly expensive,
368
        // check first for a special dataset name, which is a cheap test.
369
0
        const char *pszOptVal =
370
0
            poParentDS && strcmp(poParentDS->GetDescription(), "__debug__") == 0
371
0
                ? CPLGetConfigOption(
372
0
                      "GDAL_SIMUL_MEM_ALLOC_FAILURE_NODATA_MASK_BAND", "NO")
373
0
                : "NO";
374
0
        const bool bSimulMemAllocFailure =
375
0
            EQUAL(pszOptVal, "ALWAYS") ||
376
0
            (CPLTestBool(pszOptVal) &&
377
0
             GDALMajorObject::GetMetadataItem(__func__, "__INTERNAL__") ==
378
0
                 nullptr);
379
0
        void *pTemp = nullptr;
380
0
        if (!bSimulMemAllocFailure)
381
0
        {
382
0
            CPLErrorStateBackuper oErrorStateBackuper(CPLQuietErrorHandler);
383
0
            pTemp = VSI_MALLOC3_VERBOSE(nWrkDTSize, nBufXSize, nBufYSize);
384
0
        }
385
0
        if (!pTemp)
386
0
        {
387
0
            const bool bAllocHasAlreadyFailed =
388
0
                GDALMajorObject::GetMetadataItem(__func__, "__INTERNAL__") !=
389
0
                nullptr;
390
0
            CPLError(bAllocHasAlreadyFailed ? CE_Failure : CE_Warning,
391
0
                     CPLE_OutOfMemory,
392
0
                     "GDALNoDataMaskBand::IRasterIO(): cannot allocate %d x %d "
393
0
                     "x %d bytes%s",
394
0
                     nBufXSize, nBufYSize, nWrkDTSize,
395
0
                     bAllocHasAlreadyFailed
396
0
                         ? ""
397
0
                         : ". Falling back to block-based approach");
398
0
            if (bAllocHasAlreadyFailed)
399
0
                return std::pair(CE_Failure, nullptr);
400
            // Sets a metadata item to prevent potential infinite recursion
401
0
            GDALMajorObject::SetMetadataItem(__func__, "IN", "__INTERNAL__");
402
0
            const CPLErr eErr = GDALRasterBand::IRasterIO(
403
0
                eRWFlag, nXOff, nYOff, nXSize, nYSize, pData, nBufXSize,
404
0
                nBufYSize, eBufType, nPixelSpace, nLineSpace, psExtraArg);
405
0
            GDALMajorObject::SetMetadataItem(__func__, nullptr, "__INTERNAL__");
406
0
            return std::pair(eErr, nullptr);
407
0
        }
408
0
        return std::pair(CE_None, pTemp);
409
0
    };
410
411
0
    if (eBufType == GDT_Byte)
412
0
    {
413
0
        const int nWrkDTSize = GDALGetDataTypeSizeBytes(eWrkDT);
414
0
        auto [eErr, pTemp] = AllocTempBufferOrFallback(nWrkDTSize);
415
0
        if (!pTemp)
416
0
            return eErr;
417
418
0
        eErr = m_poParent->RasterIO(
419
0
            GF_Read, nXOff, nYOff, nXSize, nYSize, pTemp, nBufXSize, nBufYSize,
420
0
            eWrkDT, nWrkDTSize, static_cast<GSpacing>(nBufXSize) * nWrkDTSize,
421
0
            psExtraArg);
422
0
        if (eErr != CE_None)
423
0
        {
424
0
            VSIFree(pTemp);
425
0
            return eErr;
426
0
        }
427
428
0
        const bool bIsNoDataNan = std::isnan(m_dfNoDataValue) != 0;
429
0
        GByte *pabyDest = static_cast<GByte *>(pData);
430
431
        /* --------------------------------------------------------------------
432
         */
433
        /*      Process different cases. */
434
        /* --------------------------------------------------------------------
435
         */
436
0
        switch (eWrkDT)
437
0
        {
438
0
            case GDT_Byte:
439
0
            {
440
0
                const auto nNoData = static_cast<GByte>(m_dfNoDataValue);
441
0
                const auto *panSrc = static_cast<const GByte *>(pTemp);
442
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
443
0
                             nPixelSpace, nLineSpace, nNoData);
444
0
            }
445
0
            break;
446
447
0
            case GDT_Int16:
448
0
            {
449
0
                const auto nNoData = static_cast<int16_t>(m_dfNoDataValue);
450
0
                const auto *panSrc = static_cast<const int16_t *>(pTemp);
451
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
452
0
                             nPixelSpace, nLineSpace, nNoData);
453
0
            }
454
0
            break;
455
456
0
            case GDT_UInt16:
457
0
            {
458
0
                const auto nNoData = static_cast<uint16_t>(m_dfNoDataValue);
459
0
                const auto *panSrc = static_cast<const uint16_t *>(pTemp);
460
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
461
0
                             nPixelSpace, nLineSpace, nNoData);
462
0
            }
463
0
            break;
464
465
0
            case GDT_UInt32:
466
0
            {
467
0
                const auto nNoData = static_cast<GUInt32>(m_dfNoDataValue);
468
0
                const auto *panSrc = static_cast<const GUInt32 *>(pTemp);
469
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
470
0
                             nPixelSpace, nLineSpace, nNoData);
471
0
            }
472
0
            break;
473
474
0
            case GDT_Int32:
475
0
            {
476
0
                const auto nNoData = static_cast<GInt32>(m_dfNoDataValue);
477
0
                const auto *panSrc = static_cast<const GInt32 *>(pTemp);
478
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
479
0
                             nPixelSpace, nLineSpace, nNoData);
480
0
            }
481
0
            break;
482
483
0
            case GDT_Float32:
484
0
            {
485
0
                const float fNoData = static_cast<float>(m_dfNoDataValue);
486
0
                const float *pafSrc = static_cast<const float *>(pTemp);
487
488
0
                size_t i = 0;
489
0
                for (int iY = 0; iY < nBufYSize; iY++)
490
0
                {
491
0
                    GByte *pabyLineDest = pabyDest + iY * nLineSpace;
492
0
                    for (int iX = 0; iX < nBufXSize; iX++)
493
0
                    {
494
0
                        const float fVal = pafSrc[i];
495
0
                        if (bIsNoDataNan && std::isnan(fVal))
496
0
                            *pabyLineDest = 0;
497
0
                        else if (ARE_REAL_EQUAL(fVal, fNoData))
498
0
                            *pabyLineDest = 0;
499
0
                        else
500
0
                            *pabyLineDest = 255;
501
0
                        ++i;
502
0
                        pabyLineDest += nPixelSpace;
503
0
                    }
504
0
                }
505
0
            }
506
0
            break;
507
508
0
            case GDT_Float64:
509
0
            {
510
0
                const double *padfSrc = static_cast<const double *>(pTemp);
511
512
0
                size_t i = 0;
513
0
                for (int iY = 0; iY < nBufYSize; iY++)
514
0
                {
515
0
                    GByte *pabyLineDest = pabyDest + iY * nLineSpace;
516
0
                    for (int iX = 0; iX < nBufXSize; iX++)
517
0
                    {
518
0
                        const double dfVal = padfSrc[i];
519
0
                        if (bIsNoDataNan && std::isnan(dfVal))
520
0
                            *pabyLineDest = 0;
521
0
                        else if (ARE_REAL_EQUAL(dfVal, m_dfNoDataValue))
522
0
                            *pabyLineDest = 0;
523
0
                        else
524
0
                            *pabyLineDest = 255;
525
0
                        ++i;
526
0
                        pabyLineDest += nPixelSpace;
527
0
                    }
528
0
                }
529
0
            }
530
0
            break;
531
532
0
            case GDT_Int64:
533
0
            {
534
0
                const auto *panSrc = static_cast<const int64_t *>(pTemp);
535
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
536
0
                             nPixelSpace, nLineSpace, m_nNoDataValueInt64);
537
0
            }
538
0
            break;
539
540
0
            case GDT_UInt64:
541
0
            {
542
0
                const auto *panSrc = static_cast<const uint64_t *>(pTemp);
543
0
                SetZeroOr255(pabyDest, panSrc, nBufXSize, nBufYSize,
544
0
                             nPixelSpace, nLineSpace, m_nNoDataValueUInt64);
545
0
            }
546
0
            break;
547
548
0
            default:
549
0
                CPLAssert(false);
550
0
                break;
551
0
        }
552
553
0
        VSIFree(pTemp);
554
0
        return CE_None;
555
0
    }
556
557
    // Output buffer is non-Byte. Ask for Byte and expand to user requested
558
    // type
559
0
    auto [eErr, pTemp] = AllocTempBufferOrFallback(sizeof(GByte));
560
0
    if (!pTemp)
561
0
        return eErr;
562
563
0
    eErr = IRasterIO(eRWFlag, nXOff, nYOff, nXSize, nYSize, pTemp, nBufXSize,
564
0
                     nBufYSize, GDT_Byte, 1, nBufXSize, psExtraArg);
565
0
    if (eErr != CE_None)
566
0
    {
567
0
        VSIFree(pTemp);
568
0
        return eErr;
569
0
    }
570
571
0
    for (int iY = 0; iY < nBufYSize; iY++)
572
0
    {
573
0
        GDALCopyWords64(
574
0
            static_cast<GByte *>(pTemp) + static_cast<size_t>(iY) * nBufXSize,
575
0
            GDT_Byte, 1, static_cast<GByte *>(pData) + iY * nLineSpace,
576
0
            eBufType, static_cast<int>(nPixelSpace), nBufXSize);
577
0
    }
578
0
    VSIFree(pTemp);
579
0
    return CE_None;
580
0
}
581
582
/************************************************************************/
583
/*                   EmitErrorMessageIfWriteNotSupported()              */
584
/************************************************************************/
585
586
bool GDALNoDataMaskBand::EmitErrorMessageIfWriteNotSupported(
587
    const char *pszCaller) const
588
0
{
589
0
    ReportError(CE_Failure, CPLE_NoWriteAccess,
590
0
                "%s: attempt to write to a nodata implicit mask band.",
591
0
                pszCaller);
592
593
0
    return true;
594
0
}
595
596
//! @endcond