LCOV - code coverage report
Current view: top level - autotest/cpp - test_gdal.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 2499 2544 98.2 %
Date: 2025-01-18 12:42:00 Functions: 312 316 98.7 %

          Line data    Source code
       1             : ///////////////////////////////////////////////////////////////////////////////
       2             : //
       3             : // Project:  C++ Test Suite for GDAL/OGR
       4             : // Purpose:  Test general GDAL features.
       5             : // Author:   Mateusz Loskot <mateusz@loskot.net>
       6             : //
       7             : ///////////////////////////////////////////////////////////////////////////////
       8             : // Copyright (c) 2006, Mateusz Loskot <mateusz@loskot.net>
       9             : /*
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "gdal_unit_test.h"
      14             : 
      15             : #include "gdal_alg.h"
      16             : #include "gdal_priv.h"
      17             : #include "gdal_utils.h"
      18             : #include "gdal_priv_templates.hpp"
      19             : #include "gdal.h"
      20             : #include "tilematrixset.hpp"
      21             : #include "gdalcachedpixelaccessor.h"
      22             : 
      23             : #include <limits>
      24             : #include <string>
      25             : 
      26             : #include "test_data.h"
      27             : 
      28             : #include "gtest_include.h"
      29             : 
      30             : namespace
      31             : {
      32             : // Common fixture with test data
      33             : struct test_gdal : public ::testing::Test
      34             : {
      35             : };
      36             : 
      37             : // Test GDAL driver manager access
      38           4 : TEST_F(test_gdal, driver_manager)
      39             : {
      40           1 :     GDALDriverManager *drv_mgr = nullptr;
      41           1 :     drv_mgr = GetGDALDriverManager();
      42           1 :     ASSERT_TRUE(nullptr != drv_mgr);
      43             : }
      44             : 
      45             : // Test that GDALRegisterPlugins can be called
      46           4 : TEST_F(test_gdal, register_plugins)
      47             : {
      48           1 :     GDALRegisterPlugins();
      49           1 : }
      50             : 
      51             : // Test that GDALRegisterPlugin can be called and returns an error for a non
      52             : // existing plugin name
      53           4 : TEST_F(test_gdal, register_plugin)
      54             : {
      55           1 :     ASSERT_EQ(GDALRegisterPlugin("rtbreg_non_existing_plugin"), CE_Failure);
      56             : }
      57             : 
      58             : // Test number of registered GDAL drivers
      59           4 : TEST_F(test_gdal, number_of_registered_drivers)
      60             : {
      61             : #ifdef WIN32CE
      62             :     ASSERT_EQ(GDALGetDriverCount(), drv_count_);
      63             : #endif
      64           1 : }
      65             : 
      66             : // Test if AAIGrid driver is registered
      67           4 : TEST_F(test_gdal, aaigrid_is_registered)
      68             : {
      69           1 :     GDALDriverH drv = GDALGetDriverByName("AAIGrid");
      70             : 
      71             : #ifdef FRMT_aaigrid
      72             :     ASSERT_TRUE(NULL != drv);
      73             : #else
      74             :     (void)drv;
      75             : #endif
      76           1 : }
      77             : 
      78             : // Test if DTED driver is registered
      79           4 : TEST_F(test_gdal, dted_is_registered)
      80             : {
      81           1 :     GDALDriverH drv = GDALGetDriverByName("DTED");
      82             : 
      83             : #ifdef FRMT_dted
      84             :     ASSERT_TRUE(NULL != drv);
      85             : #else
      86             :     (void)drv;
      87             : #endif
      88           1 : }
      89             : 
      90             : // Test if GeoTIFF driver is registered
      91           4 : TEST_F(test_gdal, gtiff_is_registered)
      92             : {
      93           1 :     GDALDriverH drv = GDALGetDriverByName("GTiff");
      94             : 
      95             : #ifdef FRMT_gtiff
      96             :     ASSERT_TRUE(NULL != drv);
      97             : #else
      98             :     (void)drv;
      99             : #endif
     100           1 : }
     101             : 
     102             : class DataTypeTupleFixture : public test_gdal,
     103             :                              public ::testing::WithParamInterface<
     104             :                                  std::tuple<GDALDataType, GDALDataType>>
     105             : {
     106             :   public:
     107           1 :     static std::vector<std::tuple<GDALDataType, GDALDataType>> GetTupleValues()
     108             :     {
     109           1 :         std::vector<std::tuple<GDALDataType, GDALDataType>> ret;
     110          15 :         for (GDALDataType eIn = GDT_Byte; eIn < GDT_TypeCount;
     111          14 :              eIn = static_cast<GDALDataType>(eIn + 1))
     112             :         {
     113         210 :             for (GDALDataType eOut = GDT_Byte; eOut < GDT_TypeCount;
     114         196 :                  eOut = static_cast<GDALDataType>(eOut + 1))
     115             :             {
     116         196 :                 ret.emplace_back(std::make_tuple(eIn, eOut));
     117             :             }
     118             :         }
     119           1 :         return ret;
     120             :     }
     121             : };
     122             : 
     123             : // Test GDALDataTypeUnion() on all (GDALDataType, GDALDataType) combinations
     124         393 : TEST_P(DataTypeTupleFixture, GDALDataTypeUnion_generic)
     125             : {
     126         196 :     GDALDataType eDT1 = std::get<0>(GetParam());
     127         196 :     GDALDataType eDT2 = std::get<1>(GetParam());
     128         196 :     GDALDataType eDT = GDALDataTypeUnion(eDT1, eDT2);
     129         196 :     EXPECT_EQ(eDT, GDALDataTypeUnion(eDT2, eDT1));
     130         196 :     EXPECT_GE(GDALGetDataTypeSize(eDT), GDALGetDataTypeSize(eDT1));
     131         196 :     EXPECT_GE(GDALGetDataTypeSize(eDT), GDALGetDataTypeSize(eDT2));
     132         196 :     EXPECT_TRUE((GDALDataTypeIsComplex(eDT) && (GDALDataTypeIsComplex(eDT1) ||
     133             :                                                 GDALDataTypeIsComplex(eDT2))) ||
     134             :                 (!GDALDataTypeIsComplex(eDT) && !GDALDataTypeIsComplex(eDT1) &&
     135             :                  !GDALDataTypeIsComplex(eDT2)));
     136             : 
     137         196 :     EXPECT_TRUE(
     138             :         !(GDALDataTypeIsFloating(eDT1) || GDALDataTypeIsFloating(eDT2)) ||
     139             :         GDALDataTypeIsFloating(eDT));
     140         196 :     EXPECT_TRUE(!(GDALDataTypeIsSigned(eDT1) || GDALDataTypeIsSigned(eDT2)) ||
     141             :                 GDALDataTypeIsSigned(eDT));
     142         196 : }
     143             : 
     144         590 : INSTANTIATE_TEST_SUITE_P(
     145             :     test_gdal, DataTypeTupleFixture,
     146             :     ::testing::ValuesIn(DataTypeTupleFixture::GetTupleValues()),
     147             :     [](const ::testing::TestParamInfo<DataTypeTupleFixture::ParamType> &l_info)
     148             :     {
     149             :         GDALDataType eDT1 = std::get<0>(l_info.param);
     150             :         GDALDataType eDT2 = std::get<1>(l_info.param);
     151             :         return std::string(GDALGetDataTypeName(eDT1)) + "_" +
     152             :                GDALGetDataTypeName(eDT2);
     153             :     });
     154             : 
     155             : // Test GDALDataTypeUnion()
     156           4 : TEST_F(test_gdal, GDALDataTypeUnion_special_cases)
     157             : {
     158           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_Int16, GDT_UInt16), GDT_Int32);
     159           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_Int16, GDT_UInt32), GDT_Int64);
     160           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_UInt32, GDT_Int16), GDT_Int64);
     161           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_Int64, GDT_UInt64), GDT_Float64);
     162           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_Int64, GDT_Float32), GDT_Float64);
     163           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_Int64, GDT_Float64), GDT_Float64);
     164           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_UInt64, GDT_Float32), GDT_Float64);
     165           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_UInt64, GDT_Float64), GDT_Float64);
     166           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_UInt32, GDT_CInt16), GDT_CFloat64);
     167           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_Float32, GDT_CInt32), GDT_CFloat64);
     168           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt16, GDT_UInt32), GDT_CFloat64);
     169           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt16, GDT_CFloat32), GDT_CFloat32);
     170           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_Byte), GDT_CInt32);
     171           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_UInt16), GDT_CInt32);
     172           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_Int16), GDT_CInt32);
     173           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_UInt32), GDT_CFloat64);
     174           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_Int32), GDT_CInt32);
     175           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_Float32), GDT_CFloat64);
     176           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_CInt16), GDT_CInt32);
     177           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CInt32, GDT_CFloat32), GDT_CFloat64);
     178           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_Byte), GDT_CFloat32);
     179           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_UInt16), GDT_CFloat32);
     180           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_Int16), GDT_CFloat32);
     181           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_UInt32), GDT_CFloat64);
     182           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_Int32), GDT_CFloat64);
     183           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_Float32), GDT_CFloat32);
     184           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_CInt16), GDT_CFloat32);
     185           1 :     EXPECT_EQ(GDALDataTypeUnion(GDT_CFloat32, GDT_CInt32), GDT_CFloat64);
     186             : 
     187           1 :     EXPECT_EQ(GDALFindDataType(0, false /* signed */, false /* floating */,
     188             :                                false /* complex */),
     189             :               GDT_Byte);
     190           1 :     EXPECT_EQ(GDALFindDataType(0, true /* signed */, false /* floating */,
     191             :                                false /* complex */),
     192             :               GDT_Int8);
     193           1 :     EXPECT_EQ(GDALFindDataType(0, false /* signed */, false /* floating */,
     194             :                                true /* complex */),
     195             :               GDT_CInt32);
     196           1 :     EXPECT_EQ(GDALFindDataType(0, true /* signed */, false /* floating */,
     197             :                                true /* complex */),
     198             :               GDT_CInt16);
     199           1 :     EXPECT_EQ(GDALFindDataType(0, false /* signed */, true /* floating */,
     200             :                                false /* complex */),
     201             :               GDT_Float32);
     202           1 :     EXPECT_EQ(GDALFindDataType(0, true /* signed */, true /* floating */,
     203             :                                false /* complex */),
     204             :               GDT_Float32);
     205           1 :     EXPECT_EQ(GDALFindDataType(0, false /* signed */, true /* floating */,
     206             :                                true /* complex */),
     207             :               GDT_CFloat32);
     208           1 :     EXPECT_EQ(GDALFindDataType(0, true /* signed */, true /* floating */,
     209             :                                true /* complex */),
     210             :               GDT_CFloat32);
     211             : 
     212           1 :     EXPECT_EQ(GDALFindDataType(8, false /* signed */, false /* floating */,
     213             :                                false /* complex */),
     214             :               GDT_Byte);
     215           1 :     EXPECT_EQ(GDALFindDataType(8, true /* signed */, false /* floating */,
     216             :                                false /* complex */),
     217             :               GDT_Int8);
     218             : 
     219           1 :     EXPECT_EQ(GDALFindDataType(16, false /* signed */, false /* floating */,
     220             :                                false /* complex */),
     221             :               GDT_UInt16);
     222           1 :     EXPECT_EQ(GDALFindDataType(16, true /* signed */, false /* floating */,
     223             :                                false /* complex */),
     224             :               GDT_Int16);
     225             : 
     226           1 :     EXPECT_EQ(GDALFindDataType(32, false /* signed */, false /* floating */,
     227             :                                false /* complex */),
     228             :               GDT_UInt32);
     229           1 :     EXPECT_EQ(GDALFindDataType(32, true /* signed */, false /* floating */,
     230             :                                false /* complex */),
     231             :               GDT_Int32);
     232             : 
     233           1 :     EXPECT_EQ(GDALFindDataType(64, false /* signed */, true /* floating */,
     234             :                                false /* complex */),
     235             :               GDT_Float64);
     236           1 :     EXPECT_EQ(GDALFindDataType(64, false /* signed */, true /* floating */,
     237             :                                true /* complex */),
     238             :               GDT_CFloat64);
     239             : 
     240           1 :     EXPECT_EQ(GDALFindDataType(64, false /* signed */, false /* floating */,
     241             :                                false /* complex */),
     242             :               GDT_UInt64);
     243           1 :     EXPECT_EQ(GDALFindDataType(64, true /* signed */, false /* floating */,
     244             :                                false /* complex */),
     245             :               GDT_Int64);
     246             : 
     247           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Byte, -128, false), GDT_Int16);
     248           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Byte, -32768, false), GDT_Int16);
     249           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Byte, -32769, false), GDT_Int32);
     250             : 
     251           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Int8, 127, false), GDT_Int8);
     252           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Int8, 128, false), GDT_Int16);
     253             : 
     254           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Int16, 32767, false), GDT_Int16);
     255           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Int16, 32768, false), GDT_Int32);
     256             : 
     257           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_UInt16, 65535, false), GDT_UInt16);
     258           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_UInt16, 65536, false), GDT_UInt32);
     259             : 
     260           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Int32, INT32_MAX, false),
     261             :               GDT_Int32);
     262           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Int32, INT32_MAX + 1.0, false),
     263             :               GDT_Int64);
     264             : 
     265           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_UInt32, UINT32_MAX, false),
     266             :               GDT_UInt32);
     267           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_UInt32, UINT32_MAX + 1.0, false),
     268             :               GDT_UInt64);
     269             : 
     270             :     // (1 << 63) - 1024
     271           1 :     EXPECT_EQ(
     272             :         GDALDataTypeUnionWithValue(GDT_Int64, 9223372036854774784.0, false),
     273             :         GDT_Int64);
     274             :     // (1 << 63) - 512
     275           1 :     EXPECT_EQ(
     276             :         GDALDataTypeUnionWithValue(GDT_Int64, 9223372036854775296.0, false),
     277             :         GDT_Float64);
     278             : 
     279             :     // (1 << 64) - 2048
     280           1 :     EXPECT_EQ(
     281             :         GDALDataTypeUnionWithValue(GDT_UInt64, 18446744073709549568.0, false),
     282             :         GDT_UInt64);
     283             :     // (1 << 64) + 4096
     284           1 :     EXPECT_EQ(
     285             :         GDALDataTypeUnionWithValue(GDT_UInt64, 18446744073709555712.0, false),
     286             :         GDT_Float64);
     287             : 
     288           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Float32, -99999, false),
     289             :               GDT_Float32);
     290           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Float32, -99999.9876, false),
     291             :               GDT_Float64);
     292           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(
     293             :                   GDT_Float32, std::numeric_limits<double>::quiet_NaN(), false),
     294             :               GDT_Float32);
     295           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(
     296             :                   GDT_Float32, -std::numeric_limits<double>::infinity(), false),
     297             :               GDT_Float32);
     298           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(
     299             :                   GDT_Float32, -std::numeric_limits<double>::infinity(), false),
     300             :               GDT_Float32);
     301             : 
     302           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Float64, -99999.9876, false),
     303             :               GDT_Float64);
     304           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(
     305             :                   GDT_Float64, std::numeric_limits<double>::quiet_NaN(), false),
     306             :               GDT_Float64);
     307           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(
     308             :                   GDT_Float64, -std::numeric_limits<double>::infinity(), false),
     309             :               GDT_Float64);
     310           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(
     311             :                   GDT_Float64, -std::numeric_limits<double>::infinity(), false),
     312             :               GDT_Float64);
     313             : 
     314           1 :     EXPECT_EQ(GDALDataTypeUnionWithValue(GDT_Unknown, 0, false), GDT_Byte);
     315           1 : }
     316             : 
     317             : // Test GDALAdjustValueToDataType()
     318           4 : TEST_F(test_gdal, GDALAdjustValueToDataType)
     319             : {
     320             :     int bClamped, bRounded;
     321             : 
     322           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Byte, 255.0, nullptr, nullptr) ==
     323             :                 255.0);
     324           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Byte, 255.0, &bClamped,
     325             :                                           &bRounded) == 255.0 &&
     326             :                 !bClamped && !bRounded);
     327           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Byte, 254.4, &bClamped,
     328             :                                           &bRounded) == 254.0 &&
     329             :                 !bClamped && bRounded);
     330           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Byte, -1, &bClamped, &bRounded) ==
     331             :                     0.0 &&
     332             :                 bClamped && !bRounded);
     333           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Byte, 256.0, &bClamped,
     334             :                                           &bRounded) == 255.0 &&
     335             :                 bClamped && !bRounded);
     336             : 
     337           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int8, -128.0, &bClamped,
     338             :                                           &bRounded) == -128.0 &&
     339             :                 !bClamped && !bRounded);
     340           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int8, 127.0, &bClamped,
     341             :                                           &bRounded) == 127.0 &&
     342             :                 !bClamped && !bRounded);
     343           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int8, -127.4, &bClamped,
     344             :                                           &bRounded) == -127.0 &&
     345             :                 !bClamped && bRounded);
     346           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int8, 126.4, &bClamped,
     347             :                                           &bRounded) == 126.0 &&
     348             :                 !bClamped && bRounded);
     349           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int8, -129.0, &bClamped,
     350             :                                           &bRounded) == -128.0 &&
     351             :                 bClamped && !bRounded);
     352           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int8, 128.0, &bClamped,
     353             :                                           &bRounded) == 127.0 &&
     354             :                 bClamped && !bRounded);
     355             : 
     356           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt16, 65535.0, &bClamped,
     357             :                                           &bRounded) == 65535.0 &&
     358             :                 !bClamped && !bRounded);
     359           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt16, 65534.4, &bClamped,
     360             :                                           &bRounded) == 65534.0 &&
     361             :                 !bClamped && bRounded);
     362           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt16, -1, &bClamped,
     363             :                                           &bRounded) == 0.0 &&
     364             :                 bClamped && !bRounded);
     365           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt16, 65536.0, &bClamped,
     366             :                                           &bRounded) == 65535.0 &&
     367             :                 bClamped && !bRounded);
     368             : 
     369           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int16, -32768.0, &bClamped,
     370             :                                           &bRounded) == -32768.0 &&
     371             :                 !bClamped && !bRounded);
     372           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int16, 32767.0, &bClamped,
     373             :                                           &bRounded) == 32767.0 &&
     374             :                 !bClamped && !bRounded);
     375           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int16, -32767.4, &bClamped,
     376             :                                           &bRounded) == -32767.0 &&
     377             :                 !bClamped && bRounded);
     378           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int16, 32766.4, &bClamped,
     379             :                                           &bRounded) == 32766.0 &&
     380             :                 !bClamped && bRounded);
     381           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int16, -32769.0, &bClamped,
     382             :                                           &bRounded) == -32768.0 &&
     383             :                 bClamped && !bRounded);
     384           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int16, 32768.0, &bClamped,
     385             :                                           &bRounded) == 32767.0 &&
     386             :                 bClamped && !bRounded);
     387             : 
     388           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt32, 10000000.0, &bClamped,
     389             :                                           &bRounded) == 10000000.0 &&
     390             :                 !bClamped && !bRounded);
     391           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt32, 10000000.4, &bClamped,
     392             :                                           &bRounded) == 10000000.0 &&
     393             :                 !bClamped && bRounded);
     394           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt32, -1, &bClamped,
     395             :                                           &bRounded) == 0.0 &&
     396             :                 bClamped && !bRounded);
     397             : 
     398           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int32, -10000000.0, &bClamped,
     399             :                                           &bRounded) == -10000000.0 &&
     400             :                 !bClamped && !bRounded);
     401           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int32, 10000000.0, &bClamped,
     402             :                                           &bRounded) == 10000000.0 &&
     403             :                 !bClamped && !bRounded);
     404             : 
     405           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt64, 10000000000.0, &bClamped,
     406             :                                           &bRounded) == 10000000000.0 &&
     407             :                 !bClamped && !bRounded);
     408           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt64, 10000000000.4, &bClamped,
     409             :                                           &bRounded) == 10000000000.0 &&
     410             :                 !bClamped && bRounded);
     411           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_UInt64, -1, &bClamped,
     412             :                                           &bRounded) == 0.0 &&
     413             :                 bClamped && !bRounded);
     414             : 
     415           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int64, -10000000000.0, &bClamped,
     416             :                                           &bRounded) == -10000000000.0 &&
     417             :                 !bClamped && !bRounded);
     418           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Int64, 10000000000.0, &bClamped,
     419             :                                           &bRounded) == 10000000000.0 &&
     420             :                 !bClamped && !bRounded);
     421             : 
     422           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Float32, 0.0, &bClamped,
     423             :                                           &bRounded) == 0.0 &&
     424             :                 !bClamped && !bRounded);
     425           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Float32, 1e-50, &bClamped,
     426             :                                           &bRounded) == 0.0 &&
     427             :                 !bClamped && !bRounded);
     428           1 :     EXPECT_TRUE(
     429             :         GDALAdjustValueToDataType(GDT_Float32, 1.23, &bClamped, &bRounded) ==
     430             :             static_cast<double>(1.23f) &&
     431             :         !bClamped && !bRounded);
     432           1 :     EXPECT_TRUE(
     433             :         GDALAdjustValueToDataType(GDT_Float32, -1e300, &bClamped, &bRounded) ==
     434             :             -std::numeric_limits<float>::max() &&
     435             :         bClamped && !bRounded);
     436           1 :     EXPECT_TRUE(
     437             :         GDALAdjustValueToDataType(GDT_Float32, 1e300, &bClamped, &bRounded) ==
     438             :             std::numeric_limits<float>::max() &&
     439             :         bClamped && !bRounded);
     440           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(
     441             :                     GDT_Float32, std::numeric_limits<float>::infinity(),
     442             :                     &bClamped,
     443             :                     &bRounded) == std::numeric_limits<float>::infinity() &&
     444             :                 !bClamped && !bRounded);
     445           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(
     446             :                     GDT_Float32, -std::numeric_limits<float>::infinity(),
     447             :                     &bClamped,
     448             :                     &bRounded) == -std::numeric_limits<float>::infinity() &&
     449             :                 !bClamped && !bRounded);
     450             :     {
     451           1 :         double dfNan = std::numeric_limits<double>::quiet_NaN();
     452             :         double dfGot =
     453           1 :             GDALAdjustValueToDataType(GDT_Float32, dfNan, &bClamped, &bRounded);
     454           1 :         EXPECT_TRUE(memcmp(&dfNan, &dfGot, sizeof(double)) == 0 && !bClamped &&
     455             :                     !bRounded);
     456             :     }
     457             : 
     458           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Float64, 0.0, &bClamped,
     459             :                                           &bRounded) == 0.0 &&
     460             :                 !bClamped && !bRounded);
     461           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Float64, 1e-50, &bClamped,
     462             :                                           &bRounded) == 1e-50 &&
     463             :                 !bClamped && !bRounded);
     464           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Float64, -1e40, &bClamped,
     465             :                                           &bRounded) == -1e40 &&
     466             :                 !bClamped && !bRounded);
     467           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(GDT_Float64, 1e40, &bClamped,
     468             :                                           &bRounded) == 1e40 &&
     469             :                 !bClamped && !bRounded);
     470           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(
     471             :                     GDT_Float64, std::numeric_limits<float>::infinity(),
     472             :                     &bClamped,
     473             :                     &bRounded) == std::numeric_limits<float>::infinity() &&
     474             :                 !bClamped && !bRounded);
     475           1 :     EXPECT_TRUE(GDALAdjustValueToDataType(
     476             :                     GDT_Float64, -std::numeric_limits<float>::infinity(),
     477             :                     &bClamped,
     478             :                     &bRounded) == -std::numeric_limits<float>::infinity() &&
     479             :                 !bClamped && !bRounded);
     480             :     {
     481           1 :         double dfNan = std::numeric_limits<double>::quiet_NaN();
     482             :         double dfGot =
     483           1 :             GDALAdjustValueToDataType(GDT_Float64, dfNan, &bClamped, &bRounded);
     484           1 :         EXPECT_TRUE(memcmp(&dfNan, &dfGot, sizeof(double)) == 0 && !bClamped &&
     485             :                     !bRounded);
     486             :     }
     487           1 : }
     488             : 
     489             : class FakeBand : public GDALRasterBand
     490             : {
     491             :   protected:
     492           0 :     virtual CPLErr IReadBlock(int, int, void *) override
     493             :     {
     494           0 :         return CE_None;
     495             :     }
     496             : 
     497           1 :     virtual CPLErr IWriteBlock(int, int, void *) override
     498             :     {
     499           1 :         return CE_None;
     500             :     }
     501             : 
     502             :   public:
     503           1 :     FakeBand(int nXSize, int nYSize)
     504           1 :     {
     505           1 :         nBlockXSize = nXSize;
     506           1 :         nBlockYSize = nYSize;
     507           1 :     }
     508             : };
     509             : 
     510             : class DatasetWithErrorInFlushCache : public GDALDataset
     511             : {
     512             :     bool bHasFlushCache;
     513             : 
     514             :   public:
     515           2 :     DatasetWithErrorInFlushCache() : bHasFlushCache(false)
     516             :     {
     517           2 :     }
     518             : 
     519           4 :     ~DatasetWithErrorInFlushCache()
     520           2 :     {
     521           2 :         FlushCache(true);
     522           4 :     }
     523             : 
     524           4 :     virtual CPLErr FlushCache(bool bAtClosing) override
     525             :     {
     526           4 :         CPLErr eErr = CE_None;
     527           4 :         if (!bHasFlushCache)
     528             :         {
     529           2 :             CPLError(CE_Failure, CPLE_AppDefined, "some error");
     530           2 :             eErr = CE_Failure;
     531             :         }
     532           4 :         if (GDALDataset::FlushCache(bAtClosing) != CE_None)
     533           0 :             eErr = CE_Failure;
     534           4 :         bHasFlushCache = true;
     535           4 :         return eErr;
     536             :     }
     537             : 
     538           1 :     CPLErr SetSpatialRef(const OGRSpatialReference *) override
     539             :     {
     540           1 :         return CE_None;
     541             :     }
     542             : 
     543           1 :     virtual CPLErr SetGeoTransform(double *) override
     544             :     {
     545           1 :         return CE_None;
     546             :     }
     547             : 
     548           1 :     static GDALDataset *CreateCopy(const char *, GDALDataset *, int, char **,
     549             :                                    GDALProgressFunc, void *)
     550             :     {
     551           1 :         return new DatasetWithErrorInFlushCache();
     552             :     }
     553             : 
     554           1 :     static GDALDataset *Create(const char *, int nXSize, int nYSize, int,
     555             :                                GDALDataType, char **)
     556             :     {
     557           1 :         DatasetWithErrorInFlushCache *poDS = new DatasetWithErrorInFlushCache();
     558           1 :         poDS->eAccess = GA_Update;
     559           1 :         poDS->nRasterXSize = nXSize;
     560           1 :         poDS->nRasterYSize = nYSize;
     561           1 :         poDS->SetBand(1, new FakeBand(nXSize, nYSize));
     562           1 :         return poDS;
     563             :     }
     564             : };
     565             : 
     566             : // Test that GDALTranslate() detects error in flush cache
     567           4 : TEST_F(test_gdal, GDALTranslate_error_flush_cache)
     568             : {
     569           1 :     GDALDriver *poDriver = new GDALDriver();
     570           1 :     poDriver->SetDescription("DatasetWithErrorInFlushCache");
     571           1 :     poDriver->pfnCreateCopy = DatasetWithErrorInFlushCache::CreateCopy;
     572           1 :     GetGDALDriverManager()->RegisterDriver(poDriver);
     573           1 :     const char *args[] = {"-of", "DatasetWithErrorInFlushCache", nullptr};
     574             :     GDALTranslateOptions *psOptions =
     575           1 :         GDALTranslateOptionsNew((char **)args, nullptr);
     576           1 :     GDALDatasetH hSrcDS = GDALOpen(GCORE_DATA_DIR "byte.tif", GA_ReadOnly);
     577           1 :     CPLErrorReset();
     578           1 :     CPLPushErrorHandler(CPLQuietErrorHandler);
     579           1 :     GDALDatasetH hOutDS = GDALTranslate("", hSrcDS, psOptions, nullptr);
     580           1 :     CPLPopErrorHandler();
     581           1 :     GDALClose(hSrcDS);
     582           1 :     GDALTranslateOptionsFree(psOptions);
     583           1 :     EXPECT_TRUE(hOutDS == nullptr);
     584           1 :     EXPECT_TRUE(CPLGetLastErrorType() != CE_None);
     585           1 :     GetGDALDriverManager()->DeregisterDriver(poDriver);
     586           1 :     delete poDriver;
     587           1 : }
     588             : 
     589             : // Test that GDALWarp() detects error in flush cache
     590           4 : TEST_F(test_gdal, GDALWarp_error_flush_cache)
     591             : {
     592           1 :     GDALDriver *poDriver = new GDALDriver();
     593           1 :     poDriver->SetDescription("DatasetWithErrorInFlushCache");
     594           1 :     poDriver->pfnCreate = DatasetWithErrorInFlushCache::Create;
     595           1 :     GetGDALDriverManager()->RegisterDriver(poDriver);
     596           1 :     const char *args[] = {"-of", "DatasetWithErrorInFlushCache", nullptr};
     597             :     GDALWarpAppOptions *psOptions =
     598           1 :         GDALWarpAppOptionsNew((char **)args, nullptr);
     599           1 :     GDALDatasetH hSrcDS = GDALOpen(GCORE_DATA_DIR "byte.tif", GA_ReadOnly);
     600           1 :     CPLErrorReset();
     601           1 :     CPLPushErrorHandler(CPLQuietErrorHandler);
     602             :     GDALDatasetH hOutDS =
     603           1 :         GDALWarp("/", nullptr, 1, &hSrcDS, psOptions, nullptr);
     604           1 :     CPLPopErrorHandler();
     605           1 :     GDALClose(hSrcDS);
     606           1 :     GDALWarpAppOptionsFree(psOptions);
     607           1 :     EXPECT_TRUE(hOutDS == nullptr);
     608           1 :     EXPECT_TRUE(CPLGetLastErrorType() != CE_None);
     609           1 :     GetGDALDriverManager()->DeregisterDriver(poDriver);
     610           1 :     delete poDriver;
     611           1 : }
     612             : 
     613             : // Test GDALWarp() to VRT and that we can call GDALReleaseDataset() on the
     614             : // source dataset when we want.
     615           4 : TEST_F(test_gdal, GDALWarp_VRT)
     616             : {
     617           1 :     auto hDrv = GDALGetDriverByName("GTiff");
     618           1 :     if (!hDrv)
     619             :     {
     620           0 :         GTEST_SKIP() << "GTiff driver missing";
     621             :     }
     622           1 :     const char *args[] = {"-of", "VRT", nullptr};
     623             :     GDALWarpAppOptions *psOptions =
     624           1 :         GDALWarpAppOptionsNew((char **)args, nullptr);
     625           1 :     GDALDatasetH hSrcDS = GDALOpen(GCORE_DATA_DIR "byte.tif", GA_ReadOnly);
     626           1 :     GDALDatasetH hOutDS = GDALWarp("", nullptr, 1, &hSrcDS, psOptions, nullptr);
     627           1 :     GDALWarpAppOptionsFree(psOptions);
     628           1 :     GDALReleaseDataset(hSrcDS);
     629           1 :     EXPECT_EQ(GDALChecksumImage(GDALGetRasterBand(hOutDS, 1), 0, 0, 20, 20),
     630             :               4672);
     631           1 :     GDALReleaseDataset(hOutDS);
     632             : }
     633             : 
     634             : // Test GDALTranslate() to VRT and that we can call GDALReleaseDataset() on the
     635             : // source dataset when we want.
     636           4 : TEST_F(test_gdal, GDALTranslate_VRT)
     637             : {
     638           1 :     auto hDrv = GDALGetDriverByName("GTiff");
     639           1 :     if (!hDrv)
     640             :     {
     641           0 :         GTEST_SKIP() << "GTiff driver missing";
     642             :     }
     643           1 :     const char *args[] = {"-of", "VRT", nullptr};
     644             :     GDALTranslateOptions *psOptions =
     645           1 :         GDALTranslateOptionsNew((char **)args, nullptr);
     646           1 :     GDALDatasetH hSrcDS = GDALOpen(GCORE_DATA_DIR "byte.tif", GA_ReadOnly);
     647           1 :     GDALDatasetH hOutDS = GDALTranslate("", hSrcDS, psOptions, nullptr);
     648           1 :     GDALTranslateOptionsFree(psOptions);
     649           1 :     GDALReleaseDataset(hSrcDS);
     650           1 :     EXPECT_EQ(GDALChecksumImage(GDALGetRasterBand(hOutDS, 1), 0, 0, 20, 20),
     651             :               4672);
     652           1 :     GDALReleaseDataset(hOutDS);
     653             : }
     654             : 
     655             : // Test GDALBuildVRT() and that we can call GDALReleaseDataset() on the
     656             : // source dataset when we want.
     657           4 : TEST_F(test_gdal, GDALBuildVRT)
     658             : {
     659           1 :     auto hDrv = GDALGetDriverByName("GTiff");
     660           1 :     if (!hDrv)
     661             :     {
     662           0 :         GTEST_SKIP() << "GTiff driver missing";
     663             :     }
     664           1 :     GDALDatasetH hSrcDS = GDALOpen(GCORE_DATA_DIR "byte.tif", GA_ReadOnly);
     665             :     GDALDatasetH hOutDS =
     666           1 :         GDALBuildVRT("", 1, &hSrcDS, nullptr, nullptr, nullptr);
     667           1 :     GDALReleaseDataset(hSrcDS);
     668           1 :     EXPECT_EQ(GDALChecksumImage(GDALGetRasterBand(hOutDS, 1), 0, 0, 20, 20),
     669             :               4672);
     670           1 :     GDALReleaseDataset(hOutDS);
     671             : }
     672             : 
     673             : // Test that GDALSwapWords() with unaligned buffers
     674           4 : TEST_F(test_gdal, GDALSwapWords_unaligned_buffers)
     675             : {
     676           1 :     GByte abyBuffer[8 * 2 + 1] = {0, 1, 2, 3, 4, 5, 6, 7, 255,
     677             :                                   7, 6, 5, 4, 3, 2, 1, 0};
     678           1 :     GDALSwapWords(abyBuffer, 4, 2, 9);
     679           1 :     EXPECT_EQ(abyBuffer[0], 3);
     680           1 :     EXPECT_EQ(abyBuffer[1], 2);
     681           1 :     EXPECT_EQ(abyBuffer[2], 1);
     682           1 :     EXPECT_EQ(abyBuffer[3], 0);
     683             : 
     684           1 :     EXPECT_EQ(abyBuffer[9], 4);
     685           1 :     EXPECT_EQ(abyBuffer[10], 5);
     686           1 :     EXPECT_EQ(abyBuffer[11], 6);
     687           1 :     EXPECT_EQ(abyBuffer[12], 7);
     688           1 :     GDALSwapWords(abyBuffer, 4, 2, 9);
     689             : 
     690           1 :     GDALSwapWords(abyBuffer, 8, 2, 9);
     691           1 :     EXPECT_EQ(abyBuffer[0], 7);
     692           1 :     EXPECT_EQ(abyBuffer[1], 6);
     693           1 :     EXPECT_EQ(abyBuffer[2], 5);
     694           1 :     EXPECT_EQ(abyBuffer[3], 4);
     695           1 :     EXPECT_EQ(abyBuffer[4], 3);
     696           1 :     EXPECT_EQ(abyBuffer[5], 2);
     697           1 :     EXPECT_EQ(abyBuffer[6], 1);
     698           1 :     EXPECT_EQ(abyBuffer[7], 0);
     699             : 
     700           1 :     EXPECT_EQ(abyBuffer[9], 0);
     701           1 :     EXPECT_EQ(abyBuffer[10], 1);
     702           1 :     EXPECT_EQ(abyBuffer[11], 2);
     703           1 :     EXPECT_EQ(abyBuffer[12], 3);
     704           1 :     EXPECT_EQ(abyBuffer[13], 4);
     705           1 :     EXPECT_EQ(abyBuffer[14], 5);
     706           1 :     EXPECT_EQ(abyBuffer[15], 6);
     707           1 :     EXPECT_EQ(abyBuffer[16], 7);
     708           1 :     GDALSwapWords(abyBuffer, 4, 2, 9);
     709           1 : }
     710             : 
     711             : // Test ARE_REAL_EQUAL()
     712           4 : TEST_F(test_gdal, ARE_REAL_EQUAL)
     713             : {
     714           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(0.0, 0.0));
     715           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(0.0, 0.1));
     716           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(0.1, 0.0));
     717           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(1.0, 1.0));
     718           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(1.0, 0.99));
     719           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<double>::min(),
     720             :                                -std::numeric_limits<double>::min()));
     721           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(std::numeric_limits<double>::min(),
     722             :                                std::numeric_limits<double>::min()));
     723           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(std::numeric_limits<double>::min(), 0.0));
     724           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<double>::max(),
     725             :                                -std::numeric_limits<double>::max()));
     726           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(std::numeric_limits<double>::max(),
     727             :                                std::numeric_limits<double>::max()));
     728           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<double>::infinity(),
     729             :                                -std::numeric_limits<double>::infinity()));
     730           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(std::numeric_limits<double>::infinity(),
     731             :                                std::numeric_limits<double>::infinity()));
     732           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(std::numeric_limits<double>::infinity(),
     733             :                                 std::numeric_limits<double>::max()));
     734           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<double>::min(),
     735             :                                -std::numeric_limits<double>::min()));
     736             : 
     737           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(0.0f, 0.0f));
     738           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(0.0f, 0.1f));
     739           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(0.1f, 0.0f));
     740           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(1.0f, 1.0f));
     741           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(1.0f, 0.99f));
     742           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<float>::min(),
     743             :                                -std::numeric_limits<float>::min()));
     744           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(std::numeric_limits<float>::min(),
     745             :                                std::numeric_limits<float>::min()));
     746           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(std::numeric_limits<float>::min(), 0.0f));
     747           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<float>::max(),
     748             :                                -std::numeric_limits<float>::max()));
     749           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(std::numeric_limits<float>::max(),
     750             :                                std::numeric_limits<float>::max()));
     751           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(-std::numeric_limits<float>::infinity(),
     752             :                                -std::numeric_limits<float>::infinity()));
     753           1 :     EXPECT_TRUE(ARE_REAL_EQUAL(std::numeric_limits<float>::infinity(),
     754             :                                std::numeric_limits<float>::infinity()));
     755           1 :     EXPECT_TRUE(!ARE_REAL_EQUAL(std::numeric_limits<float>::infinity(),
     756             :                                 std::numeric_limits<float>::max()));
     757           1 : }
     758             : 
     759             : // Test GDALIsValueInRange()
     760           4 : TEST_F(test_gdal, GDALIsValueInRange)
     761             : {
     762           1 :     EXPECT_TRUE(GDALIsValueInRange<GByte>(0));
     763           1 :     EXPECT_TRUE(GDALIsValueInRange<GByte>(255));
     764           1 :     EXPECT_TRUE(!GDALIsValueInRange<GByte>(-1));
     765           1 :     EXPECT_TRUE(!GDALIsValueInRange<GByte>(256));
     766           1 :     EXPECT_TRUE(
     767             :         !GDALIsValueInRange<GByte>(std::numeric_limits<double>::quiet_NaN()));
     768             : 
     769           1 :     EXPECT_TRUE(GDALIsValueInRange<GInt8>(-128));
     770           1 :     EXPECT_TRUE(GDALIsValueInRange<GInt8>(127));
     771           1 :     EXPECT_TRUE(!GDALIsValueInRange<GInt8>(-129));
     772           1 :     EXPECT_TRUE(!GDALIsValueInRange<GInt8>(128));
     773             : 
     774             :     // -(1 << 63)
     775           1 :     EXPECT_TRUE(GDALIsValueInRange<int64_t>(-9223372036854775808.0));
     776             :     // (1 << 63) - 1024
     777           1 :     EXPECT_TRUE(GDALIsValueInRange<int64_t>(9223372036854774784.0));
     778           1 :     EXPECT_TRUE(GDALIsValueInRange<int64_t>(0.5));
     779             :     // (1 << 63) - 512
     780           1 :     EXPECT_TRUE(!GDALIsValueInRange<int64_t>(9223372036854775296.0));
     781             : 
     782           1 :     EXPECT_TRUE(GDALIsValueInRange<uint64_t>(0.0));
     783           1 :     EXPECT_TRUE(GDALIsValueInRange<uint64_t>(0.5));
     784             :     // (1 << 64) - 2048
     785           1 :     EXPECT_TRUE(GDALIsValueInRange<uint64_t>(18446744073709549568.0));
     786             :     // (1 << 64)
     787           1 :     EXPECT_TRUE(!GDALIsValueInRange<uint64_t>(18446744073709551616.0));
     788           1 :     EXPECT_TRUE(!GDALIsValueInRange<uint64_t>(-0.5));
     789             : 
     790           1 :     EXPECT_TRUE(GDALIsValueInRange<float>(-std::numeric_limits<float>::max()));
     791           1 :     EXPECT_TRUE(GDALIsValueInRange<float>(std::numeric_limits<float>::max()));
     792           1 :     EXPECT_TRUE(
     793             :         GDALIsValueInRange<float>(-std::numeric_limits<float>::infinity()));
     794           1 :     EXPECT_TRUE(
     795             :         GDALIsValueInRange<float>(std::numeric_limits<float>::infinity()));
     796           1 :     EXPECT_TRUE(
     797             :         !GDALIsValueInRange<float>(std::numeric_limits<double>::quiet_NaN()));
     798           1 :     EXPECT_TRUE(
     799             :         !GDALIsValueInRange<float>(-std::numeric_limits<double>::max()));
     800           1 :     EXPECT_TRUE(!GDALIsValueInRange<float>(std::numeric_limits<double>::max()));
     801             : 
     802           1 :     EXPECT_TRUE(
     803             :         GDALIsValueInRange<double>(-std::numeric_limits<double>::infinity()));
     804           1 :     EXPECT_TRUE(
     805             :         GDALIsValueInRange<double>(std::numeric_limits<double>::infinity()));
     806           1 :     EXPECT_TRUE(
     807             :         GDALIsValueInRange<double>(-std::numeric_limits<double>::max()));
     808           1 :     EXPECT_TRUE(GDALIsValueInRange<double>(std::numeric_limits<double>::max()));
     809           1 :     EXPECT_TRUE(
     810             :         !GDALIsValueInRange<double>(std::numeric_limits<double>::quiet_NaN()));
     811           1 : }
     812             : 
     813             : #ifdef _MSC_VER
     814             : #pragma warning(push)
     815             : // overflow in constant arithmetic
     816             : #pragma warning(disable : 4756)
     817             : #endif
     818             : 
     819             : // Test GDALIsValueExactAs()
     820           4 : TEST_F(test_gdal, GDALIsValueExactAs)
     821             : {
     822           1 :     EXPECT_TRUE(GDALIsValueExactAs<GByte>(0));
     823           1 :     EXPECT_TRUE(GDALIsValueExactAs<GByte>(255));
     824           1 :     EXPECT_TRUE(!GDALIsValueExactAs<GByte>(0.5));
     825           1 :     EXPECT_TRUE(!GDALIsValueExactAs<GByte>(-1));
     826           1 :     EXPECT_TRUE(!GDALIsValueExactAs<GByte>(-0.5));
     827           1 :     EXPECT_TRUE(!GDALIsValueExactAs<GByte>(255.5));
     828           1 :     EXPECT_TRUE(!GDALIsValueExactAs<GByte>(256));
     829           1 :     EXPECT_TRUE(
     830             :         !GDALIsValueExactAs<GByte>(std::numeric_limits<double>::quiet_NaN()));
     831             : 
     832             :     // -(1 << 63)
     833           1 :     EXPECT_TRUE(GDALIsValueExactAs<int64_t>(-9223372036854775808.0));
     834             :     // (1 << 63) - 1024
     835           1 :     EXPECT_TRUE(GDALIsValueExactAs<int64_t>(9223372036854774784.0));
     836           1 :     EXPECT_TRUE(!GDALIsValueExactAs<int64_t>(0.5));
     837             :     // (1 << 63) - 512
     838           1 :     EXPECT_TRUE(!GDALIsValueExactAs<int64_t>(9223372036854775296.0));
     839             : 
     840           1 :     EXPECT_TRUE(GDALIsValueExactAs<uint64_t>(0.0));
     841           1 :     EXPECT_TRUE(!GDALIsValueExactAs<uint64_t>(0.5));
     842             :     // (1 << 64) - 2048
     843           1 :     EXPECT_TRUE(GDALIsValueExactAs<uint64_t>(18446744073709549568.0));
     844             :     // (1 << 64)
     845           1 :     EXPECT_TRUE(!GDALIsValueExactAs<uint64_t>(18446744073709551616.0));
     846           1 :     EXPECT_TRUE(!GDALIsValueExactAs<uint64_t>(-0.5));
     847             : 
     848           1 :     EXPECT_TRUE(GDALIsValueExactAs<float>(-std::numeric_limits<float>::max()));
     849           1 :     EXPECT_TRUE(GDALIsValueExactAs<float>(std::numeric_limits<float>::max()));
     850           1 :     EXPECT_TRUE(
     851             :         GDALIsValueExactAs<float>(-std::numeric_limits<float>::infinity()));
     852           1 :     EXPECT_TRUE(
     853             :         GDALIsValueExactAs<float>(std::numeric_limits<float>::infinity()));
     854           1 :     EXPECT_TRUE(
     855             :         GDALIsValueExactAs<float>(std::numeric_limits<double>::quiet_NaN()));
     856           1 :     EXPECT_TRUE(
     857             :         !GDALIsValueExactAs<float>(-std::numeric_limits<double>::max()));
     858           1 :     EXPECT_TRUE(!GDALIsValueExactAs<float>(std::numeric_limits<double>::max()));
     859             : 
     860           1 :     EXPECT_TRUE(
     861             :         GDALIsValueExactAs<double>(-std::numeric_limits<double>::infinity()));
     862           1 :     EXPECT_TRUE(
     863             :         GDALIsValueExactAs<double>(std::numeric_limits<double>::infinity()));
     864           1 :     EXPECT_TRUE(
     865             :         GDALIsValueExactAs<double>(-std::numeric_limits<double>::max()));
     866           1 :     EXPECT_TRUE(GDALIsValueExactAs<double>(std::numeric_limits<double>::max()));
     867           1 :     EXPECT_TRUE(
     868             :         GDALIsValueExactAs<double>(std::numeric_limits<double>::quiet_NaN()));
     869           1 : }
     870             : 
     871             : // Test GDALIsValueExactAs()
     872           4 : TEST_F(test_gdal, GDALIsValueExactAs_C_func)
     873             : {
     874           1 :     EXPECT_TRUE(GDALIsValueExactAs(0, GDT_Byte));
     875           1 :     EXPECT_TRUE(GDALIsValueExactAs(255, GDT_Byte));
     876           1 :     EXPECT_FALSE(GDALIsValueExactAs(-1, GDT_Byte));
     877           1 :     EXPECT_FALSE(GDALIsValueExactAs(256, GDT_Byte));
     878           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_Byte));
     879             : 
     880           1 :     EXPECT_TRUE(GDALIsValueExactAs(-128, GDT_Int8));
     881           1 :     EXPECT_TRUE(GDALIsValueExactAs(127, GDT_Int8));
     882           1 :     EXPECT_FALSE(GDALIsValueExactAs(-129, GDT_Int8));
     883           1 :     EXPECT_FALSE(GDALIsValueExactAs(128, GDT_Int8));
     884           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_Int8));
     885             : 
     886           1 :     EXPECT_TRUE(GDALIsValueExactAs(0, GDT_UInt16));
     887           1 :     EXPECT_TRUE(GDALIsValueExactAs(65535, GDT_UInt16));
     888           1 :     EXPECT_FALSE(GDALIsValueExactAs(-1, GDT_UInt16));
     889           1 :     EXPECT_FALSE(GDALIsValueExactAs(65536, GDT_UInt16));
     890           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_UInt16));
     891             : 
     892           1 :     EXPECT_TRUE(GDALIsValueExactAs(-32768, GDT_Int16));
     893           1 :     EXPECT_TRUE(GDALIsValueExactAs(32767, GDT_Int16));
     894           1 :     EXPECT_FALSE(GDALIsValueExactAs(-32769, GDT_Int16));
     895           1 :     EXPECT_FALSE(GDALIsValueExactAs(32768, GDT_Int16));
     896           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_Int16));
     897             : 
     898           1 :     EXPECT_TRUE(GDALIsValueExactAs(std::numeric_limits<uint32_t>::lowest(),
     899             :                                    GDT_UInt32));
     900           1 :     EXPECT_TRUE(
     901             :         GDALIsValueExactAs(std::numeric_limits<uint32_t>::max(), GDT_UInt32));
     902           1 :     EXPECT_FALSE(GDALIsValueExactAs(
     903             :         std::numeric_limits<uint32_t>::lowest() - 1.0, GDT_UInt32));
     904           1 :     EXPECT_FALSE(GDALIsValueExactAs(std::numeric_limits<uint32_t>::max() + 1.0,
     905             :                                     GDT_UInt32));
     906           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_UInt32));
     907             : 
     908           1 :     EXPECT_TRUE(
     909             :         GDALIsValueExactAs(std::numeric_limits<int32_t>::lowest(), GDT_Int32));
     910           1 :     EXPECT_TRUE(
     911             :         GDALIsValueExactAs(std::numeric_limits<int32_t>::max(), GDT_Int32));
     912           1 :     EXPECT_FALSE(GDALIsValueExactAs(
     913             :         std::numeric_limits<int32_t>::lowest() - 1.0, GDT_Int32));
     914           1 :     EXPECT_FALSE(GDALIsValueExactAs(std::numeric_limits<int32_t>::max() + 1.0,
     915             :                                     GDT_Int32));
     916           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_Int32));
     917             : 
     918           1 :     EXPECT_TRUE(GDALIsValueExactAs(
     919             :         static_cast<double>(std::numeric_limits<uint64_t>::lowest()),
     920             :         GDT_UInt64));
     921             :     // (1 << 64) - 2048
     922           1 :     EXPECT_TRUE(GDALIsValueExactAs(18446744073709549568.0, GDT_UInt64));
     923           1 :     EXPECT_FALSE(GDALIsValueExactAs(
     924             :         static_cast<double>(std::numeric_limits<uint64_t>::lowest()) - 1.0,
     925             :         GDT_UInt64));
     926             :     // (1 << 64)
     927           1 :     EXPECT_FALSE(GDALIsValueExactAs(18446744073709551616.0, GDT_UInt64));
     928           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_UInt64));
     929             : 
     930           1 :     EXPECT_TRUE(GDALIsValueExactAs(
     931             :         static_cast<double>(std::numeric_limits<int64_t>::lowest()),
     932             :         GDT_Int64));
     933             :     // (1 << 63) - 1024
     934           1 :     EXPECT_TRUE(GDALIsValueExactAs(9223372036854774784.0, GDT_Int64));
     935           1 :     EXPECT_FALSE(GDALIsValueExactAs(
     936             :         static_cast<double>(std::numeric_limits<int64_t>::lowest()) - 2048.0,
     937             :         GDT_Int64));
     938             :     // (1 << 63) - 512
     939           1 :     EXPECT_FALSE(GDALIsValueExactAs(9223372036854775296.0, GDT_Int64));
     940           1 :     EXPECT_FALSE(GDALIsValueExactAs(0.5, GDT_Int64));
     941             : 
     942           1 :     EXPECT_TRUE(
     943             :         GDALIsValueExactAs(-std::numeric_limits<float>::max(), GDT_Float32));
     944           1 :     EXPECT_TRUE(
     945             :         GDALIsValueExactAs(std::numeric_limits<float>::max(), GDT_Float32));
     946           1 :     EXPECT_TRUE(GDALIsValueExactAs(-std::numeric_limits<float>::infinity(),
     947             :                                    GDT_Float32));
     948           1 :     EXPECT_TRUE(GDALIsValueExactAs(std::numeric_limits<float>::infinity(),
     949             :                                    GDT_Float32));
     950           1 :     EXPECT_TRUE(GDALIsValueExactAs(std::numeric_limits<double>::quiet_NaN(),
     951             :                                    GDT_Float32));
     952           1 :     EXPECT_TRUE(
     953             :         !GDALIsValueExactAs(-std::numeric_limits<double>::max(), GDT_Float32));
     954           1 :     EXPECT_TRUE(
     955             :         !GDALIsValueExactAs(std::numeric_limits<double>::max(), GDT_Float32));
     956             : 
     957           1 :     EXPECT_TRUE(GDALIsValueExactAs(-std::numeric_limits<double>::infinity(),
     958             :                                    GDT_Float64));
     959           1 :     EXPECT_TRUE(GDALIsValueExactAs(std::numeric_limits<double>::infinity(),
     960             :                                    GDT_Float64));
     961           1 :     EXPECT_TRUE(
     962             :         GDALIsValueExactAs(-std::numeric_limits<double>::max(), GDT_Float64));
     963           1 :     EXPECT_TRUE(
     964             :         GDALIsValueExactAs(std::numeric_limits<double>::max(), GDT_Float64));
     965           1 :     EXPECT_TRUE(GDALIsValueExactAs(std::numeric_limits<double>::quiet_NaN(),
     966             :                                    GDT_Float64));
     967             : 
     968           1 :     EXPECT_TRUE(GDALIsValueExactAs(0, GDT_CInt16));
     969           1 : }
     970             : 
     971             : #ifdef _MSC_VER
     972             : #pragma warning(pop)
     973             : #endif
     974             : 
     975             : // Test GDALDataTypeIsInteger()
     976           4 : TEST_F(test_gdal, GDALDataTypeIsInteger)
     977             : {
     978           1 :     EXPECT_TRUE(!GDALDataTypeIsInteger(GDT_Unknown));
     979           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_Byte), TRUE);
     980           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_Int8), TRUE);
     981           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_UInt16), TRUE);
     982           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_Int16), TRUE);
     983           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_UInt32), TRUE);
     984           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_Int32), TRUE);
     985           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_UInt64), TRUE);
     986           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_Int64), TRUE);
     987           1 :     EXPECT_TRUE(!GDALDataTypeIsInteger(GDT_Float32));
     988           1 :     EXPECT_TRUE(!GDALDataTypeIsInteger(GDT_Float64));
     989           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_CInt16), TRUE);
     990           1 :     EXPECT_EQ(GDALDataTypeIsInteger(GDT_CInt32), TRUE);
     991           1 :     EXPECT_TRUE(!GDALDataTypeIsInteger(GDT_CFloat32));
     992           1 :     EXPECT_TRUE(!GDALDataTypeIsInteger(GDT_CFloat64));
     993           1 : }
     994             : 
     995             : // Test GDALDataTypeIsFloating()
     996           4 : TEST_F(test_gdal, GDALDataTypeIsFloating)
     997             : {
     998           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_Unknown));
     999           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_Byte));
    1000           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_Int8));
    1001           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_UInt16));
    1002           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_Int16));
    1003           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_UInt32));
    1004           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_Int32));
    1005           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_UInt64));
    1006           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_Int64));
    1007           1 :     EXPECT_EQ(GDALDataTypeIsFloating(GDT_Float32), TRUE);
    1008           1 :     EXPECT_EQ(GDALDataTypeIsFloating(GDT_Float64), TRUE);
    1009           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_CInt16));
    1010           1 :     EXPECT_TRUE(!GDALDataTypeIsFloating(GDT_CInt32));
    1011           1 :     EXPECT_EQ(GDALDataTypeIsFloating(GDT_CFloat32), TRUE);
    1012           1 :     EXPECT_EQ(GDALDataTypeIsFloating(GDT_CFloat64), TRUE);
    1013           1 : }
    1014             : 
    1015             : // Test GDALDataTypeIsComplex()
    1016           4 : TEST_F(test_gdal, GDALDataTypeIsComplex)
    1017             : {
    1018           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Unknown));
    1019           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Byte));
    1020           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Int8));
    1021           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_UInt16));
    1022           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Int16));
    1023           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_UInt32));
    1024           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Int32));
    1025           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_UInt64));
    1026           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Int64));
    1027           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Float32));
    1028           1 :     EXPECT_TRUE(!GDALDataTypeIsComplex(GDT_Float64));
    1029           1 :     EXPECT_EQ(GDALDataTypeIsComplex(GDT_CInt16), TRUE);
    1030           1 :     EXPECT_EQ(GDALDataTypeIsComplex(GDT_CInt32), TRUE);
    1031           1 :     EXPECT_EQ(GDALDataTypeIsComplex(GDT_CFloat32), TRUE);
    1032           1 :     EXPECT_EQ(GDALDataTypeIsComplex(GDT_CFloat64), TRUE);
    1033           1 : }
    1034             : 
    1035             : // Test GDALDataTypeIsConversionLossy()
    1036           4 : TEST_F(test_gdal, GDALDataTypeIsConversionLossy)
    1037             : {
    1038           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Byte));
    1039           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Int8));
    1040           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_UInt16));
    1041           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Int16));
    1042           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_UInt32));
    1043           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Int32));
    1044           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_UInt64));
    1045           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Int64));
    1046           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Float32));
    1047           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_Float64));
    1048           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_CInt16));
    1049           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_CInt32));
    1050           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_CFloat32));
    1051           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Byte, GDT_CFloat64));
    1052             : 
    1053           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Byte));
    1054           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Int8));
    1055           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int8, GDT_UInt16));
    1056           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Int16));
    1057           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int8, GDT_UInt32));
    1058           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Int32));
    1059           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int8, GDT_UInt64));
    1060           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Int64));
    1061           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Float32));
    1062           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_Float64));
    1063           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_CInt16));
    1064           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_CInt32));
    1065           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_CFloat32));
    1066           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int8, GDT_CFloat64));
    1067             : 
    1068           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Byte));
    1069           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Int8));
    1070           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_UInt16));
    1071           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Int16));
    1072           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_UInt32));
    1073           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Int32));
    1074           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_UInt64));
    1075           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Int64));
    1076           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Float32));
    1077           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_Float64));
    1078           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_CInt16));
    1079           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_CInt32));
    1080           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_CFloat32));
    1081           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt16, GDT_CFloat64));
    1082             : 
    1083           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Byte));
    1084           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Int8));
    1085           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int16, GDT_UInt16));
    1086           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Int16));
    1087           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int16, GDT_UInt32));
    1088           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Int32));
    1089           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int16, GDT_UInt64));
    1090           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Int64));
    1091           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Float32));
    1092           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_Float64));
    1093           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_CInt16));
    1094           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_CInt32));
    1095           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_CFloat32));
    1096           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int16, GDT_CFloat64));
    1097             : 
    1098           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_Byte));
    1099           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_UInt16));
    1100           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_Int16));
    1101           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_UInt32));
    1102           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_Int32));
    1103           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_UInt64));
    1104           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_Int64));
    1105           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_Float32));
    1106           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_Float64));
    1107           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_CInt16));
    1108           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_CInt32));
    1109           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_CFloat32));
    1110           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt32, GDT_CFloat64));
    1111             : 
    1112           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_Byte));
    1113           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_UInt16));
    1114           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_Int16));
    1115           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_UInt32));
    1116           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int32, GDT_Int32));
    1117           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_UInt64));
    1118           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int32, GDT_Int64));
    1119           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_Float32));
    1120           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int32, GDT_Float64));
    1121           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_CInt16));
    1122           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int32, GDT_CInt32));
    1123           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int32, GDT_CFloat32));
    1124           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int32, GDT_CFloat64));
    1125             : 
    1126           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_Byte));
    1127           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_UInt16));
    1128           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_Int16));
    1129           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_UInt32));
    1130           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_Int32));
    1131           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_UInt64));
    1132           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_Int64));
    1133           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_Float32));
    1134           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_Float64));
    1135           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_CInt16));
    1136           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_CInt32));
    1137           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_CFloat32));
    1138           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_UInt64, GDT_CFloat64));
    1139             : 
    1140           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_Byte));
    1141           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_UInt16));
    1142           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_Int16));
    1143           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_UInt32));
    1144           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_Int32));
    1145           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_UInt64));
    1146           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Int64, GDT_Int64));
    1147           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_Float32));
    1148           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_Float64));
    1149           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_CInt16));
    1150           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_CInt32));
    1151           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_CFloat32));
    1152           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Int64, GDT_CFloat64));
    1153             : 
    1154           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_Byte));
    1155           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_UInt16));
    1156           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_Int16));
    1157           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_UInt32));
    1158           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_Int32));
    1159           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_UInt64));
    1160           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_Int64));
    1161           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Float32, GDT_Float32));
    1162           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Float32, GDT_Float64));
    1163           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_CInt16));
    1164           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float32, GDT_CInt32));
    1165           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Float32, GDT_CFloat32));
    1166           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Float32, GDT_CFloat64));
    1167             : 
    1168           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_Byte));
    1169           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_UInt16));
    1170           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_Int16));
    1171           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_UInt32));
    1172           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_Int32));
    1173           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_UInt64));
    1174           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_Int64));
    1175           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_Float32));
    1176           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Float64, GDT_Float64));
    1177           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_CInt16));
    1178           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_CInt32));
    1179           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_Float64, GDT_CFloat32));
    1180           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_Float64, GDT_CFloat64));
    1181             : 
    1182           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_Byte));
    1183           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_UInt16));
    1184           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_Int16));
    1185           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_UInt32));
    1186           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_Int32));
    1187           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_UInt64));
    1188           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_Int64));
    1189           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_Float32));
    1190           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_Float64));
    1191           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_CInt16));
    1192           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_CInt32));
    1193           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_CFloat32));
    1194           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CInt16, GDT_CFloat64));
    1195             : 
    1196           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_Byte));
    1197           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_UInt16));
    1198           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_Int16));
    1199           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_UInt32));
    1200           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_Int32));
    1201           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_UInt64));
    1202           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_Int64));
    1203           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_Float32));
    1204           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_Float64));
    1205           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_CInt16));
    1206           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_CInt32));
    1207           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_CFloat32));
    1208           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CInt32, GDT_CFloat64));
    1209             : 
    1210           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_Byte));
    1211           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_UInt16));
    1212           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_Int16));
    1213           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_UInt32));
    1214           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_Int32));
    1215           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_UInt64));
    1216           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_Int64));
    1217           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_Float32));
    1218           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_Float64));
    1219           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_CInt16));
    1220           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_CInt32));
    1221           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_CFloat32));
    1222           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CFloat32, GDT_CFloat64));
    1223             : 
    1224           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_Byte));
    1225           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_UInt16));
    1226           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_Int16));
    1227           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_UInt32));
    1228           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_Int32));
    1229           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_UInt64));
    1230           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_Int64));
    1231           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_Float32));
    1232           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_Float64));
    1233           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_CInt16));
    1234           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_CInt32));
    1235           1 :     EXPECT_TRUE(GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_CFloat32));
    1236           1 :     EXPECT_TRUE(!GDALDataTypeIsConversionLossy(GDT_CFloat64, GDT_CFloat64));
    1237           1 : }
    1238             : 
    1239             : // Test GDALDataset::GetBands()
    1240           4 : TEST_F(test_gdal, GDALDataset_GetBands)
    1241             : {
    1242             :     GDALDatasetUniquePtr poDS(GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    1243           1 :                                   ->Create("", 1, 1, 3, GDT_Byte, nullptr));
    1244           1 :     int nExpectedNumber = 1;
    1245           4 :     for (auto &&poBand : poDS->GetBands())
    1246             :     {
    1247           3 :         EXPECT_EQ(poBand->GetBand(), nExpectedNumber);
    1248           3 :         nExpectedNumber++;
    1249             :     }
    1250           1 :     ASSERT_EQ(nExpectedNumber, 3 + 1);
    1251             : 
    1252           1 :     ASSERT_EQ(poDS->GetBands().size(), 3U);
    1253           1 :     EXPECT_EQ(poDS->GetBands()[0], poDS->GetRasterBand(1));
    1254           1 :     EXPECT_EQ(poDS->GetBands()[static_cast<size_t>(0)], poDS->GetRasterBand(1));
    1255             : }
    1256             : 
    1257           4 : TEST_F(test_gdal, GDALExtendedDataType)
    1258             : {
    1259             :     // non-null string to string
    1260             :     {
    1261           1 :         const char *srcPtr = "foo";
    1262           1 :         char *dstPtr = nullptr;
    1263           1 :         GDALExtendedDataType::CopyValue(
    1264           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &dstPtr,
    1265           2 :             GDALExtendedDataType::CreateString());
    1266           1 :         EXPECT_TRUE(dstPtr != nullptr);
    1267             :         // Coverity isn't smart enough to figure out that GetClass() of
    1268             :         // CreateString() is GEDTC_STRING and then takes the wrong path
    1269             :         // in CopyValue() and makes wrong assumptions.
    1270             :         // coverity[string_null]
    1271           1 :         EXPECT_STREQ(dstPtr, srcPtr);
    1272             :         // coverity[incorrect_free]
    1273           1 :         CPLFree(dstPtr);
    1274             :     }
    1275             :     // null string to string
    1276             :     {
    1277           1 :         const char *srcPtr = nullptr;
    1278           1 :         char *dstPtr = nullptr;
    1279           1 :         GDALExtendedDataType::CopyValue(
    1280           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &dstPtr,
    1281           2 :             GDALExtendedDataType::CreateString());
    1282           1 :         EXPECT_TRUE(dstPtr == nullptr);
    1283             :     }
    1284             :     // non-null string to Int32
    1285             :     {
    1286           1 :         const char *srcPtr = "2";
    1287           1 :         int32_t nVal = 1;
    1288           1 :         GDALExtendedDataType::CopyValue(
    1289           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &nVal,
    1290           2 :             GDALExtendedDataType::Create(GDT_Int32));
    1291           1 :         EXPECT_EQ(nVal, 2);
    1292             :     }
    1293             :     // null string to Int32
    1294             :     {
    1295           1 :         const char *srcPtr = nullptr;
    1296           1 :         int32_t nVal = 1;
    1297           1 :         GDALExtendedDataType::CopyValue(
    1298           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &nVal,
    1299           2 :             GDALExtendedDataType::Create(GDT_Int32));
    1300           1 :         EXPECT_EQ(nVal, 0);
    1301             :     }
    1302             :     // non-null string to Int64
    1303             :     {
    1304           1 :         const char *srcPtr = "2";
    1305           1 :         int64_t nVal = 1;
    1306           1 :         GDALExtendedDataType::CopyValue(
    1307           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &nVal,
    1308           2 :             GDALExtendedDataType::Create(GDT_Int64));
    1309           1 :         EXPECT_EQ(nVal, 2);
    1310             :     }
    1311             :     // null string to Int64
    1312             :     {
    1313           1 :         const char *srcPtr = nullptr;
    1314           1 :         int64_t nVal = 1;
    1315           1 :         GDALExtendedDataType::CopyValue(
    1316           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &nVal,
    1317           2 :             GDALExtendedDataType::Create(GDT_Int64));
    1318           1 :         EXPECT_EQ(nVal, 0);
    1319             :     }
    1320             :     // non-null string to UInt64
    1321             :     {
    1322           1 :         char *srcPtr = nullptr;
    1323           1 :         uint64_t nVal = 1;
    1324           1 :         GDALExtendedDataType::CopyValue(
    1325           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &nVal,
    1326           2 :             GDALExtendedDataType::Create(GDT_UInt64));
    1327           1 :         EXPECT_EQ(nVal, 0U);
    1328             :     }
    1329             :     // non-null string to Int64
    1330             :     {
    1331           1 :         const char *srcPtr = "2";
    1332           1 :         uint64_t nVal = 1;
    1333           1 :         GDALExtendedDataType::CopyValue(
    1334           2 :             &srcPtr, GDALExtendedDataType::CreateString(), &nVal,
    1335           2 :             GDALExtendedDataType::Create(GDT_UInt64));
    1336           1 :         EXPECT_EQ(nVal, 2U);
    1337             :     }
    1338             : 
    1339             :     class myArray : public GDALMDArray
    1340             :     {
    1341             :         GDALExtendedDataType m_dt;
    1342             :         std::vector<std::shared_ptr<GDALDimension>> m_dims;
    1343             :         std::vector<GUInt64> m_blockSize;
    1344             :         const std::string m_osEmptyFilename{};
    1345             : 
    1346             :         static std::vector<std::shared_ptr<GDALDimension>>
    1347           9 :         BuildDims(const std::vector<GUInt64> &sizes)
    1348             :         {
    1349           9 :             std::vector<std::shared_ptr<GDALDimension>> dims;
    1350          33 :             for (const auto sz : sizes)
    1351             :             {
    1352             :                 dims.emplace_back(
    1353          24 :                     std::make_shared<GDALDimension>("", "", "", "", sz));
    1354             :             }
    1355           9 :             return dims;
    1356             :         }
    1357             : 
    1358             :       protected:
    1359           0 :         bool IRead(const GUInt64 *, const size_t *, const GInt64 *,
    1360             :                    const GPtrDiff_t *, const GDALExtendedDataType &,
    1361             :                    void *) const override
    1362             :         {
    1363           0 :             return false;
    1364             :         }
    1365             : 
    1366             :       public:
    1367           8 :         myArray(GDALDataType eDT, const std::vector<GUInt64> &sizes,
    1368             :                 const std::vector<GUInt64> &blocksizes)
    1369           8 :             : GDALAbstractMDArray("", "array"), GDALMDArray("", "array"),
    1370             :               m_dt(GDALExtendedDataType::Create(eDT)), m_dims(BuildDims(sizes)),
    1371           8 :               m_blockSize(blocksizes)
    1372             :         {
    1373           8 :         }
    1374             : 
    1375           1 :         myArray(const GDALExtendedDataType &dt,
    1376             :                 const std::vector<GUInt64> &sizes,
    1377             :                 const std::vector<GUInt64> &blocksizes)
    1378           1 :             : GDALAbstractMDArray("", "array"), GDALMDArray("", "array"),
    1379           1 :               m_dt(dt), m_dims(BuildDims(sizes)), m_blockSize(blocksizes)
    1380             :         {
    1381           1 :         }
    1382             : 
    1383           0 :         bool IsWritable() const override
    1384             :         {
    1385           0 :             return true;
    1386             :         }
    1387             : 
    1388           0 :         const std::string &GetFilename() const override
    1389             :         {
    1390           0 :             return m_osEmptyFilename;
    1391             :         }
    1392             : 
    1393             :         static std::shared_ptr<myArray>
    1394           1 :         Create(GDALDataType eDT, const std::vector<GUInt64> &sizes,
    1395             :                const std::vector<GUInt64> &blocksizes)
    1396             :         {
    1397             :             auto ar(
    1398           1 :                 std::shared_ptr<myArray>(new myArray(eDT, sizes, blocksizes)));
    1399           1 :             ar->SetSelf(ar);
    1400           1 :             return ar;
    1401             :         }
    1402             : 
    1403             :         static std::shared_ptr<myArray>
    1404           1 :         Create(const GDALExtendedDataType &dt,
    1405             :                const std::vector<GUInt64> &sizes,
    1406             :                const std::vector<GUInt64> &blocksizes)
    1407             :         {
    1408             :             auto ar(
    1409           1 :                 std::shared_ptr<myArray>(new myArray(dt, sizes, blocksizes)));
    1410           1 :             ar->SetSelf(ar);
    1411           1 :             return ar;
    1412             :         }
    1413             : 
    1414             :         const std::vector<std::shared_ptr<GDALDimension>> &
    1415          74 :         GetDimensions() const override
    1416             :         {
    1417          74 :             return m_dims;
    1418             :         }
    1419             : 
    1420          14 :         const GDALExtendedDataType &GetDataType() const override
    1421             :         {
    1422          14 :             return m_dt;
    1423             :         }
    1424             : 
    1425          10 :         std::vector<GUInt64> GetBlockSize() const override
    1426             :         {
    1427          10 :             return m_blockSize;
    1428             :         }
    1429             :     };
    1430             : 
    1431             :     {
    1432           3 :         auto ar(myArray::Create(GDT_UInt16, {3000, 1000, 2000}, {32, 64, 128}));
    1433           1 :         EXPECT_EQ(ar->at(0)->GetDimensionCount(), 2U);
    1434           1 :         EXPECT_EQ(ar->at(2999, 999, 1999)->GetDimensionCount(), 0U);
    1435           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    1436           1 :         EXPECT_TRUE(ar->at(3000, 0, 0) == nullptr);
    1437           1 :         EXPECT_TRUE(ar->at(0, 0, 0, 0) == nullptr);
    1438           1 :         EXPECT_TRUE((*ar)["foo"] == nullptr);
    1439           1 :         CPLPopErrorHandler();
    1440             :     }
    1441             : 
    1442             :     {
    1443           2 :         std::vector<std::unique_ptr<GDALEDTComponent>> comps;
    1444             :         comps.emplace_back(
    1445           4 :             std::unique_ptr<GDALEDTComponent>(new GDALEDTComponent(
    1446           3 :                 "f\\o\"o", 0, GDALExtendedDataType::Create(GDT_Int32))));
    1447           3 :         auto dt(GDALExtendedDataType::Create("", 4, std::move(comps)));
    1448           3 :         auto ar(myArray::Create(dt, {3000, 1000, 2000}, {32, 64, 128}));
    1449           1 :         EXPECT_TRUE((*ar)["f\\o\"o"] != nullptr);
    1450             :     }
    1451             : 
    1452             :     {
    1453           3 :         myArray ar(GDT_UInt16, {}, {});
    1454             : 
    1455           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    1456           1 :         EXPECT_TRUE(ar.GetView("[...]") == nullptr);
    1457           1 :         CPLPopErrorHandler();
    1458             : 
    1459           2 :         auto cs = ar.GetProcessingChunkSize(0);
    1460           1 :         EXPECT_EQ(cs.size(), 0U);
    1461             : 
    1462             :         struct TmpStructNoDim
    1463             :         {
    1464           1 :             static bool func(GDALAbstractMDArray *p_ar,
    1465             :                              const GUInt64 *chunk_array_start_idx,
    1466             :                              const size_t *chunk_count, GUInt64 iCurChunk,
    1467             :                              GUInt64 nChunkCount, void *user_data)
    1468             :             {
    1469           1 :                 EXPECT_TRUE(p_ar->GetName() == "array");
    1470           1 :                 EXPECT_TRUE(chunk_array_start_idx == nullptr);
    1471           1 :                 EXPECT_TRUE(chunk_count == nullptr);
    1472           1 :                 EXPECT_EQ(iCurChunk, 1U);
    1473           1 :                 EXPECT_EQ(nChunkCount, 1U);
    1474           1 :                 *static_cast<bool *>(user_data) = true;
    1475           1 :                 return true;
    1476             :             }
    1477             :         };
    1478             : 
    1479           1 :         bool b = false;
    1480           1 :         ar.ProcessPerChunk(nullptr, nullptr, nullptr, TmpStructNoDim::func, &b);
    1481           1 :         EXPECT_TRUE(b);
    1482             :     }
    1483             : 
    1484             :     struct ChunkDef
    1485             :     {
    1486             :         std::vector<GUInt64> array_start_idx;
    1487             :         std::vector<GUInt64> count;
    1488             :     };
    1489             : 
    1490             :     struct TmpStruct
    1491             :     {
    1492          16 :         static bool func(GDALAbstractMDArray *p_ar,
    1493             :                          const GUInt64 *chunk_array_start_idx,
    1494             :                          const size_t *chunk_count, GUInt64 iCurChunk,
    1495             :                          GUInt64 nChunkCount, void *user_data)
    1496             :         {
    1497          16 :             EXPECT_EQ(p_ar->GetName(), "array");
    1498          16 :             std::vector<ChunkDef> *p_chunkDefs =
    1499             :                 static_cast<std::vector<ChunkDef> *>(user_data);
    1500          32 :             std::vector<GUInt64> v_chunk_array_start_idx;
    1501             :             v_chunk_array_start_idx.insert(
    1502           0 :                 v_chunk_array_start_idx.end(), chunk_array_start_idx,
    1503          16 :                 chunk_array_start_idx + p_ar->GetDimensionCount());
    1504          32 :             std::vector<GUInt64> v_chunk_count;
    1505           0 :             v_chunk_count.insert(v_chunk_count.end(), chunk_count,
    1506          16 :                                  chunk_count + p_ar->GetDimensionCount());
    1507          16 :             ChunkDef chunkDef;
    1508          16 :             chunkDef.array_start_idx = std::move(v_chunk_array_start_idx);
    1509          16 :             chunkDef.count = std::move(v_chunk_count);
    1510          16 :             p_chunkDefs->emplace_back(std::move(chunkDef));
    1511          16 :             EXPECT_EQ(p_chunkDefs->size(), iCurChunk);
    1512          16 :             EXPECT_TRUE(iCurChunk > 0);
    1513          16 :             EXPECT_TRUE(iCurChunk <= nChunkCount);
    1514          32 :             return true;
    1515             :         }
    1516             :     };
    1517             : 
    1518             :     {
    1519           3 :         myArray ar(GDT_UInt16, {3000, 1000, 2000}, {32, 64, 128});
    1520             :         {
    1521           2 :             auto cs = ar.GetProcessingChunkSize(0);
    1522           1 :             EXPECT_EQ(cs.size(), 3U);
    1523           1 :             EXPECT_EQ(cs[0], 32U);
    1524           1 :             EXPECT_EQ(cs[1], 64U);
    1525           1 :             EXPECT_EQ(cs[2], 128U);
    1526             :         }
    1527             :         {
    1528           2 :             auto cs = ar.GetProcessingChunkSize(40 * 1000 * 1000);
    1529           1 :             EXPECT_EQ(cs.size(), 3U);
    1530           1 :             EXPECT_EQ(cs[0], 32U);
    1531           1 :             EXPECT_EQ(cs[1], 256U);
    1532           1 :             EXPECT_EQ(cs[2], 2000U);
    1533             : 
    1534           2 :             std::vector<ChunkDef> chunkDefs;
    1535             : 
    1536             :             // Error cases of input parameters of ProcessPerChunk()
    1537             :             {
    1538             :                 // array_start_idx[0] + count[0] > 3000
    1539           2 :                 std::vector<GUInt64> array_start_idx{1, 0, 0};
    1540           2 :                 std::vector<GUInt64> count{3000, 1000, 2000};
    1541           1 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1542           1 :                 EXPECT_TRUE(!ar.ProcessPerChunk(array_start_idx.data(),
    1543             :                                                 count.data(), cs.data(),
    1544             :                                                 TmpStruct::func, &chunkDefs));
    1545           1 :                 CPLPopErrorHandler();
    1546             :             }
    1547             :             {
    1548             :                 // array_start_idx[0] >= 3000
    1549           2 :                 std::vector<GUInt64> array_start_idx{3000, 0, 0};
    1550           2 :                 std::vector<GUInt64> count{1, 1000, 2000};
    1551           1 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1552           1 :                 EXPECT_TRUE(!ar.ProcessPerChunk(array_start_idx.data(),
    1553             :                                                 count.data(), cs.data(),
    1554             :                                                 TmpStruct::func, &chunkDefs));
    1555           1 :                 CPLPopErrorHandler();
    1556             :             }
    1557             :             {
    1558             :                 // count[0] > 3000
    1559           2 :                 std::vector<GUInt64> array_start_idx{0, 0, 0};
    1560           2 :                 std::vector<GUInt64> count{3001, 1000, 2000};
    1561           1 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1562           1 :                 EXPECT_TRUE(!ar.ProcessPerChunk(array_start_idx.data(),
    1563             :                                                 count.data(), cs.data(),
    1564             :                                                 TmpStruct::func, &chunkDefs));
    1565           1 :                 CPLPopErrorHandler();
    1566             :             }
    1567             :             {
    1568             :                 // count[0] == 0
    1569           2 :                 std::vector<GUInt64> array_start_idx{0, 0, 0};
    1570           2 :                 std::vector<GUInt64> count{0, 1000, 2000};
    1571           1 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1572           1 :                 EXPECT_TRUE(!ar.ProcessPerChunk(array_start_idx.data(),
    1573             :                                                 count.data(), cs.data(),
    1574             :                                                 TmpStruct::func, &chunkDefs));
    1575           1 :                 CPLPopErrorHandler();
    1576             :             }
    1577             :             {
    1578             :                 // myCustomChunkSize[0] == 0
    1579           2 :                 std::vector<GUInt64> array_start_idx{0, 0, 0};
    1580           2 :                 std::vector<GUInt64> count{3000, 1000, 2000};
    1581           2 :                 std::vector<size_t> myCustomChunkSize{0, 1000, 2000};
    1582           1 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1583           1 :                 EXPECT_TRUE(!ar.ProcessPerChunk(
    1584             :                     array_start_idx.data(), count.data(),
    1585             :                     myCustomChunkSize.data(), TmpStruct::func, &chunkDefs));
    1586           1 :                 CPLPopErrorHandler();
    1587             :             }
    1588             :             {
    1589             :                 // myCustomChunkSize[0] > 3000
    1590           2 :                 std::vector<GUInt64> array_start_idx{0, 0, 0};
    1591           2 :                 std::vector<GUInt64> count{3000, 1000, 2000};
    1592           2 :                 std::vector<size_t> myCustomChunkSize{3001, 1000, 2000};
    1593           1 :                 CPLPushErrorHandler(CPLQuietErrorHandler);
    1594           1 :                 EXPECT_TRUE(!ar.ProcessPerChunk(
    1595             :                     array_start_idx.data(), count.data(),
    1596             :                     myCustomChunkSize.data(), TmpStruct::func, &chunkDefs));
    1597           1 :                 CPLPopErrorHandler();
    1598             :             }
    1599             : 
    1600           2 :             std::vector<GUInt64> array_start_idx{1500, 256, 0};
    1601           2 :             std::vector<GUInt64> count{99, 512, 2000};
    1602           1 :             EXPECT_TRUE(ar.ProcessPerChunk(array_start_idx.data(), count.data(),
    1603             :                                            cs.data(), TmpStruct::func,
    1604             :                                            &chunkDefs));
    1605             : 
    1606           1 :             size_t nExpectedChunks = 1;
    1607           4 :             for (size_t i = 0; i < ar.GetDimensionCount(); i++)
    1608             :             {
    1609           3 :                 nExpectedChunks *= static_cast<size_t>(
    1610           3 :                     1 + ((array_start_idx[i] + count[i] - 1) / cs[i]) -
    1611           3 :                     (array_start_idx[i] / cs[i]));
    1612             :             }
    1613           1 :             EXPECT_EQ(chunkDefs.size(), nExpectedChunks);
    1614             : 
    1615           2 :             CPLString osChunks;
    1616           9 :             for (const auto &chunkDef : chunkDefs)
    1617             :             {
    1618             :                 osChunks += CPLSPrintf("{%u, %u, %u}, {%u, %u, %u}\n",
    1619           8 :                                        (unsigned)chunkDef.array_start_idx[0],
    1620           8 :                                        (unsigned)chunkDef.array_start_idx[1],
    1621           8 :                                        (unsigned)chunkDef.array_start_idx[2],
    1622           8 :                                        (unsigned)chunkDef.count[0],
    1623           8 :                                        (unsigned)chunkDef.count[1],
    1624           8 :                                        (unsigned)chunkDef.count[2]);
    1625             :             }
    1626           1 :             EXPECT_EQ(osChunks, "{1500, 256, 0}, {4, 256, 2000}\n"
    1627             :                                 "{1500, 512, 0}, {4, 256, 2000}\n"
    1628             :                                 "{1504, 256, 0}, {32, 256, 2000}\n"
    1629             :                                 "{1504, 512, 0}, {32, 256, 2000}\n"
    1630             :                                 "{1536, 256, 0}, {32, 256, 2000}\n"
    1631             :                                 "{1536, 512, 0}, {32, 256, 2000}\n"
    1632             :                                 "{1568, 256, 0}, {31, 256, 2000}\n"
    1633             :                                 "{1568, 512, 0}, {31, 256, 2000}\n");
    1634             :         }
    1635             :     }
    1636             : 
    1637             :     // Another error case of ProcessPerChunk
    1638             :     {
    1639           1 :         const auto M64 = std::numeric_limits<GUInt64>::max();
    1640           1 :         const auto Msize_t = std::numeric_limits<size_t>::max();
    1641           3 :         myArray ar(GDT_UInt16, {M64, M64, M64}, {32, 256, 128});
    1642             : 
    1643             :         // Product of myCustomChunkSize[] > Msize_t
    1644           2 :         std::vector<GUInt64> array_start_idx{0, 0, 0};
    1645           2 :         std::vector<GUInt64> count{3000, 1000, 2000};
    1646           2 :         std::vector<size_t> myCustomChunkSize{Msize_t, Msize_t, Msize_t};
    1647           2 :         std::vector<ChunkDef> chunkDefs;
    1648           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    1649           1 :         EXPECT_TRUE(!ar.ProcessPerChunk(array_start_idx.data(), count.data(),
    1650             :                                         myCustomChunkSize.data(),
    1651             :                                         TmpStruct::func, &chunkDefs));
    1652           1 :         CPLPopErrorHandler();
    1653             :     }
    1654             : 
    1655             :     {
    1656           1 :         const auto BIG = GUInt64(5000) * 1000 * 1000;
    1657             :         myArray ar(GDT_UInt16, {BIG + 3000, BIG + 1000, BIG + 2000},
    1658           3 :                    {32, 256, 128});
    1659           2 :         std::vector<GUInt64> array_start_idx{BIG + 1500, BIG + 256, BIG + 0};
    1660           2 :         std::vector<GUInt64> count{99, 512, 2000};
    1661           2 :         std::vector<ChunkDef> chunkDefs;
    1662           2 :         auto cs = ar.GetProcessingChunkSize(40 * 1000 * 1000);
    1663           1 :         EXPECT_TRUE(ar.ProcessPerChunk(array_start_idx.data(), count.data(),
    1664             :                                        cs.data(), TmpStruct::func, &chunkDefs));
    1665             : 
    1666           1 :         size_t nExpectedChunks = 1;
    1667           4 :         for (size_t i = 0; i < ar.GetDimensionCount(); i++)
    1668             :         {
    1669           3 :             nExpectedChunks *= static_cast<size_t>(
    1670           3 :                 1 + ((array_start_idx[i] + count[i] - 1) / cs[i]) -
    1671           3 :                 (array_start_idx[i] / cs[i]));
    1672             :         }
    1673           1 :         EXPECT_EQ(chunkDefs.size(), nExpectedChunks);
    1674             : 
    1675           2 :         CPLString osChunks;
    1676           9 :         for (const auto &chunkDef : chunkDefs)
    1677             :         {
    1678             :             osChunks += CPLSPrintf("{" CPL_FRMT_GUIB ", " CPL_FRMT_GUIB
    1679             :                                    ", " CPL_FRMT_GUIB "}, {%u, %u, %u}\n",
    1680           8 :                                    (GUIntBig)chunkDef.array_start_idx[0],
    1681           8 :                                    (GUIntBig)chunkDef.array_start_idx[1],
    1682           8 :                                    (GUIntBig)chunkDef.array_start_idx[2],
    1683           8 :                                    (unsigned)chunkDef.count[0],
    1684           8 :                                    (unsigned)chunkDef.count[1],
    1685           8 :                                    (unsigned)chunkDef.count[2]);
    1686             :         }
    1687           1 :         EXPECT_EQ(osChunks,
    1688             :                   "{5000001500, 5000000256, 5000000000}, {4, 256, 2000}\n"
    1689             :                   "{5000001500, 5000000512, 5000000000}, {4, 256, 2000}\n"
    1690             :                   "{5000001504, 5000000256, 5000000000}, {32, 256, 2000}\n"
    1691             :                   "{5000001504, 5000000512, 5000000000}, {32, 256, 2000}\n"
    1692             :                   "{5000001536, 5000000256, 5000000000}, {32, 256, 2000}\n"
    1693             :                   "{5000001536, 5000000512, 5000000000}, {32, 256, 2000}\n"
    1694             :                   "{5000001568, 5000000256, 5000000000}, {31, 256, 2000}\n"
    1695             :                   "{5000001568, 5000000512, 5000000000}, {31, 256, 2000}\n");
    1696             :     }
    1697             : 
    1698             :     {
    1699             :         // Test with 0 in GetBlockSize()
    1700           3 :         myArray ar(GDT_UInt16, {500, 1000, 2000}, {0, 0, 128});
    1701             :         {
    1702           2 :             auto cs = ar.GetProcessingChunkSize(300 * 2);
    1703           1 :             EXPECT_EQ(cs.size(), 3U);
    1704           1 :             EXPECT_EQ(cs[0], 1U);
    1705           1 :             EXPECT_EQ(cs[1], 1U);
    1706           1 :             EXPECT_EQ(cs[2], 256U);
    1707             :         }
    1708             :         {
    1709           2 :             auto cs = ar.GetProcessingChunkSize(40 * 1000 * 1000);
    1710           1 :             EXPECT_EQ(cs.size(), 3U);
    1711           1 :             EXPECT_EQ(cs[0], 10U);
    1712           1 :             EXPECT_EQ(cs[1], 1000U);
    1713           1 :             EXPECT_EQ(cs[2], 2000U);
    1714             :         }
    1715             :         {
    1716           2 :             auto cs = ar.GetProcessingChunkSize(500U * 1000 * 2000 * 2);
    1717           1 :             EXPECT_EQ(cs.size(), 3U);
    1718           1 :             EXPECT_EQ(cs[0], 500U);
    1719           1 :             EXPECT_EQ(cs[1], 1000U);
    1720           1 :             EXPECT_EQ(cs[2], 2000U);
    1721             :         }
    1722             :         {
    1723           2 :             auto cs = ar.GetProcessingChunkSize(500U * 1000 * 2000 * 2 - 1);
    1724           1 :             EXPECT_EQ(cs.size(), 3U);
    1725           1 :             EXPECT_EQ(cs[0], 499U);
    1726           1 :             EXPECT_EQ(cs[1], 1000U);
    1727           1 :             EXPECT_EQ(cs[2], 2000U);
    1728             :         }
    1729             :     }
    1730             :     {
    1731           1 :         const auto M = std::numeric_limits<GUInt64>::max();
    1732           3 :         myArray ar(GDT_UInt16, {M, M, M}, {M, M, M / 2});
    1733             :         {
    1734           2 :             auto cs = ar.GetProcessingChunkSize(0);
    1735           1 :             EXPECT_EQ(cs.size(), 3U);
    1736           1 :             EXPECT_EQ(cs[0], 1U);
    1737           1 :             EXPECT_EQ(cs[1], 1U);
    1738             : #if SIZEOF_VOIDP == 8
    1739           1 :             EXPECT_EQ(cs[2], static_cast<size_t>(M / 2));
    1740             : #else
    1741             :             EXPECT_EQ(cs[2], 1U);
    1742             : #endif
    1743             :         }
    1744             :     }
    1745             : #if SIZEOF_VOIDP == 8
    1746             :     {
    1747           1 :         const auto M = std::numeric_limits<GUInt64>::max();
    1748           3 :         myArray ar(GDT_UInt16, {M, M, M}, {M, M, M / 4});
    1749             :         {
    1750             :             auto cs =
    1751           2 :                 ar.GetProcessingChunkSize(std::numeric_limits<size_t>::max());
    1752           1 :             EXPECT_EQ(cs.size(), 3U);
    1753           1 :             EXPECT_EQ(cs[0], 1U);
    1754           1 :             EXPECT_EQ(cs[1], 1U);
    1755           1 :             EXPECT_EQ(cs[2], (std::numeric_limits<size_t>::max() / 4) * 2);
    1756             :         }
    1757             :     }
    1758             : #endif
    1759           1 : }
    1760             : 
    1761             : // Test GDALDataset::GetRawBinaryLayout() implementations
    1762           4 : TEST_F(test_gdal, GetRawBinaryLayout_ENVI)
    1763             : {
    1764           1 :     if (GDALGetDriverByName("ENVI") == nullptr)
    1765             :     {
    1766           0 :         GTEST_SKIP() << "ENVI driver missing";
    1767             :     }
    1768             : 
    1769             :     {
    1770             :         GDALDatasetUniquePtr poDS(
    1771           2 :             GDALDataset::Open(GDRIVERS_DATA_DIR "envi/envi_rgbsmall_bip.img"));
    1772           1 :         EXPECT_TRUE(poDS != nullptr);
    1773           2 :         GDALDataset::RawBinaryLayout sLayout;
    1774           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1775           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1776           1 :         EXPECT_EQ(
    1777             :             static_cast<int>(sLayout.eInterleaving),
    1778             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BIP));
    1779           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    1780           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    1781           1 :         EXPECT_EQ(sLayout.nImageOffset, 0U);
    1782           1 :         EXPECT_EQ(sLayout.nPixelOffset, 3);
    1783           1 :         EXPECT_EQ(sLayout.nLineOffset, 3 * 50);
    1784           1 :         EXPECT_EQ(sLayout.nBandOffset, 1);
    1785             :     }
    1786             : 
    1787             :     {
    1788             :         GDALDatasetUniquePtr poDS(
    1789           2 :             GDALDataset::Open(GDRIVERS_DATA_DIR "envi/envi_rgbsmall_bil.img"));
    1790           1 :         EXPECT_TRUE(poDS != nullptr);
    1791           2 :         GDALDataset::RawBinaryLayout sLayout;
    1792           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1793           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1794           1 :         EXPECT_EQ(
    1795             :             static_cast<int>(sLayout.eInterleaving),
    1796             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BIL));
    1797           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    1798           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    1799           1 :         EXPECT_EQ(sLayout.nImageOffset, 0U);
    1800           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    1801           1 :         EXPECT_EQ(sLayout.nLineOffset, 3 * 50);
    1802           1 :         EXPECT_EQ(sLayout.nBandOffset, 50);
    1803             :     }
    1804             : 
    1805             :     {
    1806             :         GDALDatasetUniquePtr poDS(
    1807           2 :             GDALDataset::Open(GDRIVERS_DATA_DIR "envi/envi_rgbsmall_bsq.img"));
    1808           1 :         EXPECT_TRUE(poDS != nullptr);
    1809           2 :         GDALDataset::RawBinaryLayout sLayout;
    1810           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1811           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1812           1 :         EXPECT_EQ(
    1813             :             static_cast<int>(sLayout.eInterleaving),
    1814             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BSQ));
    1815           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    1816           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    1817           1 :         EXPECT_EQ(sLayout.nImageOffset, 0U);
    1818           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    1819           1 :         EXPECT_EQ(sLayout.nLineOffset, 50);
    1820           1 :         EXPECT_EQ(sLayout.nBandOffset, 50 * 49);
    1821             :     }
    1822             : }
    1823             : 
    1824             : // Test GDALDataset::GetRawBinaryLayout() implementations
    1825           4 : TEST_F(test_gdal, GetRawBinaryLayout_GTIFF)
    1826             : {
    1827           1 :     if (GDALGetDriverByName("GTIFF") == nullptr)
    1828             :     {
    1829           0 :         GTEST_SKIP() << "GTIFF driver missing";
    1830             :     }
    1831             : 
    1832             :     {
    1833             :         GDALDatasetUniquePtr poDS(
    1834           2 :             GDALDataset::Open(GCORE_DATA_DIR "uint16.tif"));
    1835           1 :         EXPECT_TRUE(poDS != nullptr);
    1836           2 :         GDALDataset::RawBinaryLayout sLayout;
    1837           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1838           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1839           1 :         EXPECT_EQ(static_cast<int>(sLayout.eInterleaving),
    1840             :                   static_cast<int>(
    1841             :                       GDALDataset::RawBinaryLayout::Interleaving::UNKNOWN));
    1842           1 :         EXPECT_EQ(sLayout.eDataType, GDT_UInt16);
    1843           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    1844           1 :         EXPECT_EQ(sLayout.nImageOffset, 8U);
    1845           1 :         EXPECT_EQ(sLayout.nPixelOffset, 2);
    1846           1 :         EXPECT_EQ(sLayout.nLineOffset, 40);
    1847           1 :         EXPECT_EQ(sLayout.nBandOffset, 0);
    1848             :     }
    1849             : 
    1850             :     {
    1851             :         GDALDatasetUniquePtr poDS(
    1852           2 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif"));
    1853           1 :         EXPECT_TRUE(poDS != nullptr);
    1854           2 :         GDALDataset::RawBinaryLayout sLayout;
    1855             :         // Compressed
    1856           1 :         EXPECT_TRUE(!poDS->GetRawBinaryLayout(sLayout));
    1857             :     }
    1858             : 
    1859             :     {
    1860             :         GDALDatasetUniquePtr poDS(
    1861           2 :             GDALDataset::Open(GCORE_DATA_DIR "stefan_full_rgba.tif"));
    1862           1 :         EXPECT_TRUE(poDS != nullptr);
    1863           2 :         GDALDataset::RawBinaryLayout sLayout;
    1864           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1865           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1866           1 :         EXPECT_EQ(
    1867             :             static_cast<int>(sLayout.eInterleaving),
    1868             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BIP));
    1869           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    1870           1 :         EXPECT_EQ(sLayout.nImageOffset, 278U);
    1871           1 :         EXPECT_EQ(sLayout.nPixelOffset, 4);
    1872           1 :         EXPECT_EQ(sLayout.nLineOffset, 162 * 4);
    1873           1 :         EXPECT_EQ(sLayout.nBandOffset, 1);
    1874             :     }
    1875             : 
    1876             :     {
    1877             :         GDALDatasetUniquePtr poSrcDS(
    1878           2 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif"));
    1879           1 :         EXPECT_TRUE(poSrcDS != nullptr);
    1880           1 :         auto tmpFilename = "/vsimem/tmp.tif";
    1881           1 :         auto poDrv = GDALDriver::FromHandle(GDALGetDriverByName("GTiff"));
    1882           1 :         const char *options[] = {"INTERLEAVE=BAND", nullptr};
    1883             :         auto poDS(GDALDatasetUniquePtr(
    1884             :             poDrv->CreateCopy(tmpFilename, poSrcDS.get(), false,
    1885           2 :                               const_cast<char **>(options), nullptr, nullptr)));
    1886           1 :         EXPECT_TRUE(poDS != nullptr);
    1887           2 :         GDALDataset::RawBinaryLayout sLayout;
    1888           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1889           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1890           1 :         EXPECT_EQ(
    1891             :             static_cast<int>(sLayout.eInterleaving),
    1892             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BSQ));
    1893           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    1894           1 :         EXPECT_TRUE(sLayout.nImageOffset >= 396U);
    1895           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    1896           1 :         EXPECT_EQ(sLayout.nLineOffset, 50);
    1897           1 :         EXPECT_EQ(sLayout.nBandOffset, 50 * 50);
    1898           1 :         poDS.reset();
    1899           1 :         VSIUnlink(tmpFilename);
    1900             :     }
    1901             : 
    1902             :     {
    1903             :         GDALDatasetUniquePtr poSrcDS(
    1904           2 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif"));
    1905           1 :         EXPECT_TRUE(poSrcDS != nullptr);
    1906           1 :         auto tmpFilename = "/vsimem/tmp.tif";
    1907           1 :         const char *options[] = {"-srcwin",
    1908             :                                  "0",
    1909             :                                  "0",
    1910             :                                  "48",
    1911             :                                  "32",
    1912             :                                  "-co",
    1913             :                                  "INTERLEAVE=PIXEL",
    1914             :                                  "-co",
    1915             :                                  "TILED=YES",
    1916             :                                  "-co",
    1917             :                                  "BLOCKXSIZE=48",
    1918             :                                  "-co",
    1919             :                                  "BLOCKYSIZE=32",
    1920             :                                  nullptr};
    1921             :         auto psOptions =
    1922           1 :             GDALTranslateOptionsNew(const_cast<char **>(options), nullptr);
    1923             :         auto poDS(GDALDatasetUniquePtr(GDALDataset::FromHandle(
    1924             :             GDALTranslate(tmpFilename, GDALDataset::ToHandle(poSrcDS.get()),
    1925           2 :                           psOptions, nullptr))));
    1926           1 :         GDALTranslateOptionsFree(psOptions);
    1927           1 :         EXPECT_TRUE(poDS != nullptr);
    1928           2 :         GDALDataset::RawBinaryLayout sLayout;
    1929           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1930           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1931           1 :         EXPECT_EQ(
    1932             :             static_cast<int>(sLayout.eInterleaving),
    1933             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BIP));
    1934           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    1935           1 :         EXPECT_TRUE(sLayout.nImageOffset >= 390U);
    1936           1 :         EXPECT_EQ(sLayout.nPixelOffset, 3);
    1937           1 :         EXPECT_EQ(sLayout.nLineOffset, 48 * 3);
    1938           1 :         EXPECT_EQ(sLayout.nBandOffset, 1);
    1939           1 :         poDS.reset();
    1940           1 :         VSIUnlink(tmpFilename);
    1941             :     }
    1942             : 
    1943             :     {
    1944             :         GDALDatasetUniquePtr poSrcDS(
    1945           2 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif"));
    1946           1 :         EXPECT_TRUE(poSrcDS != nullptr);
    1947           1 :         auto tmpFilename = "/vsimem/tmp.tif";
    1948           1 :         const char *options[] = {"-srcwin",
    1949             :                                  "0",
    1950             :                                  "0",
    1951             :                                  "48",
    1952             :                                  "32",
    1953             :                                  "-ot",
    1954             :                                  "UInt16",
    1955             :                                  "-co",
    1956             :                                  "TILED=YES",
    1957             :                                  "-co",
    1958             :                                  "BLOCKXSIZE=48",
    1959             :                                  "-co",
    1960             :                                  "BLOCKYSIZE=32",
    1961             :                                  "-co",
    1962             :                                  "INTERLEAVE=BAND",
    1963             :                                  "-co",
    1964             :                                  "ENDIANNESS=BIG",
    1965             :                                  nullptr};
    1966             :         auto psOptions =
    1967           1 :             GDALTranslateOptionsNew(const_cast<char **>(options), nullptr);
    1968             :         auto poDS(GDALDatasetUniquePtr(GDALDataset::FromHandle(
    1969             :             GDALTranslate(tmpFilename, GDALDataset::ToHandle(poSrcDS.get()),
    1970           2 :                           psOptions, nullptr))));
    1971           1 :         GDALTranslateOptionsFree(psOptions);
    1972           1 :         EXPECT_TRUE(poDS != nullptr);
    1973           2 :         GDALDataset::RawBinaryLayout sLayout;
    1974           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    1975           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    1976           1 :         EXPECT_EQ(
    1977             :             static_cast<int>(sLayout.eInterleaving),
    1978             :             static_cast<int>(GDALDataset::RawBinaryLayout::Interleaving::BSQ));
    1979           1 :         EXPECT_EQ(sLayout.eDataType, GDT_UInt16);
    1980           1 :         EXPECT_TRUE(!sLayout.bLittleEndianOrder);
    1981           1 :         EXPECT_TRUE(sLayout.nImageOffset >= 408U);
    1982           1 :         EXPECT_EQ(sLayout.nPixelOffset, 2);
    1983           1 :         EXPECT_EQ(sLayout.nLineOffset, 2 * 48);
    1984           1 :         EXPECT_EQ(sLayout.nBandOffset, 2 * 48 * 32);
    1985           1 :         poDS.reset();
    1986           1 :         VSIUnlink(tmpFilename);
    1987             :     }
    1988             : }
    1989             : 
    1990             : // Test GDALDataset::GetRawBinaryLayout() implementations
    1991           4 : TEST_F(test_gdal, GetRawBinaryLayout_ISIS3)
    1992             : {
    1993           1 :     if (GDALGetDriverByName("ISIS3") == nullptr)
    1994             :     {
    1995           0 :         GTEST_SKIP() << "ISIS3 driver missing";
    1996             :     }
    1997             : 
    1998             :     {
    1999             :         GDALDatasetUniquePtr poDS(
    2000           2 :             GDALDataset::Open(GDRIVERS_DATA_DIR "isis3/isis3_detached.lbl"));
    2001           1 :         EXPECT_TRUE(poDS != nullptr);
    2002           2 :         GDALDataset::RawBinaryLayout sLayout;
    2003           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    2004           1 :         EXPECT_TRUE(sLayout.osRawFilename.find("isis3_detached.cub") !=
    2005             :                     std::string::npos);
    2006           1 :         EXPECT_EQ(static_cast<int>(sLayout.eInterleaving),
    2007             :                   static_cast<int>(
    2008             :                       GDALDataset::RawBinaryLayout::Interleaving::UNKNOWN));
    2009           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    2010           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    2011           1 :         EXPECT_EQ(sLayout.nImageOffset, 0U);
    2012           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    2013           1 :         EXPECT_EQ(sLayout.nLineOffset, 317);
    2014             :         // EXPECT_EQ( sLayout.nBandOffset, 9510 ); // doesn't matter on single
    2015             :         // band
    2016             :     }
    2017             : }
    2018             : 
    2019             : // Test GDALDataset::GetRawBinaryLayout() implementations
    2020           4 : TEST_F(test_gdal, GetRawBinaryLayout_VICAR)
    2021             : {
    2022           1 :     if (GDALGetDriverByName("VICAR") == nullptr)
    2023             :     {
    2024           0 :         GTEST_SKIP() << "VICAR driver missing";
    2025             :     }
    2026             : 
    2027             :     {
    2028             :         GDALDatasetUniquePtr poDS(GDALDataset::Open(
    2029           2 :             GDRIVERS_DATA_DIR "vicar/test_vicar_truncated.bin"));
    2030           1 :         EXPECT_TRUE(poDS != nullptr);
    2031           2 :         GDALDataset::RawBinaryLayout sLayout;
    2032           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    2033           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    2034           1 :         EXPECT_EQ(static_cast<int>(sLayout.eInterleaving),
    2035             :                   static_cast<int>(
    2036             :                       GDALDataset::RawBinaryLayout::Interleaving::UNKNOWN));
    2037           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    2038           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    2039           1 :         EXPECT_EQ(sLayout.nImageOffset, 9680U);
    2040           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    2041           1 :         EXPECT_EQ(sLayout.nLineOffset, 400);
    2042           1 :         EXPECT_EQ(sLayout.nBandOffset, 0);  // doesn't matter on single band
    2043             :     }
    2044             : }
    2045             : 
    2046             : // Test GDALDataset::GetRawBinaryLayout() implementations
    2047           4 : TEST_F(test_gdal, GetRawBinaryLayout_FITS)
    2048             : {
    2049           1 :     if (GDALGetDriverByName("FITS") == nullptr)
    2050             :     {
    2051           0 :         GTEST_SKIP() << "FITS driver missing";
    2052             :     }
    2053             : 
    2054             :     {
    2055             :         GDALDatasetUniquePtr poSrcDS(
    2056           2 :             GDALDataset::Open(GCORE_DATA_DIR "int16.tif"));
    2057           1 :         EXPECT_TRUE(poSrcDS != nullptr);
    2058           2 :         CPLString tmpFilename(CPLGenerateTempFilename(nullptr));
    2059           1 :         tmpFilename += ".fits";
    2060           1 :         auto poDrv = GDALDriver::FromHandle(GDALGetDriverByName("FITS"));
    2061           1 :         if (poDrv)
    2062             :         {
    2063             :             auto poDS(GDALDatasetUniquePtr(poDrv->CreateCopy(
    2064           2 :                 tmpFilename, poSrcDS.get(), false, nullptr, nullptr, nullptr)));
    2065           1 :             EXPECT_TRUE(poDS != nullptr);
    2066           1 :             poDS.reset();
    2067           1 :             poDS.reset(GDALDataset::Open(tmpFilename));
    2068           1 :             EXPECT_TRUE(poDS != nullptr);
    2069           2 :             GDALDataset::RawBinaryLayout sLayout;
    2070           1 :             EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    2071           1 :             EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    2072           1 :             EXPECT_EQ(static_cast<int>(sLayout.eInterleaving),
    2073             :                       static_cast<int>(
    2074             :                           GDALDataset::RawBinaryLayout::Interleaving::UNKNOWN));
    2075           1 :             EXPECT_EQ(sLayout.eDataType, GDT_Int16);
    2076           1 :             EXPECT_TRUE(!sLayout.bLittleEndianOrder);
    2077           1 :             EXPECT_EQ(sLayout.nImageOffset, 2880U);
    2078           1 :             EXPECT_EQ(sLayout.nPixelOffset, 2);
    2079           1 :             EXPECT_EQ(sLayout.nLineOffset, 2 * 20);
    2080           1 :             EXPECT_EQ(sLayout.nBandOffset, 2 * 20 * 20);
    2081           1 :             poDS.reset();
    2082           1 :             VSIUnlink(tmpFilename);
    2083             :         }
    2084             :     }
    2085             : }
    2086             : 
    2087             : // Test GDALDataset::GetRawBinaryLayout() implementations
    2088           4 : TEST_F(test_gdal, GetRawBinaryLayout_PDS)
    2089             : {
    2090           1 :     if (GDALGetDriverByName("PDS") == nullptr)
    2091             :     {
    2092           0 :         GTEST_SKIP() << "PDS driver missing";
    2093             :     }
    2094             : 
    2095             :     {
    2096             :         GDALDatasetUniquePtr poDS(
    2097           2 :             GDALDataset::Open(GDRIVERS_DATA_DIR "pds/mc02_truncated.img"));
    2098           1 :         EXPECT_TRUE(poDS != nullptr);
    2099           2 :         GDALDataset::RawBinaryLayout sLayout;
    2100           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    2101           1 :         EXPECT_EQ(sLayout.osRawFilename, poDS->GetDescription());
    2102           1 :         EXPECT_EQ(static_cast<int>(sLayout.eInterleaving),
    2103             :                   static_cast<int>(
    2104             :                       GDALDataset::RawBinaryLayout::Interleaving::UNKNOWN));
    2105           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    2106           1 :         EXPECT_TRUE(sLayout.bLittleEndianOrder);
    2107           1 :         EXPECT_EQ(sLayout.nImageOffset, 3840U);
    2108           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    2109           1 :         EXPECT_EQ(sLayout.nLineOffset, 3840);
    2110           1 :         EXPECT_EQ(sLayout.nBandOffset, 0);  // doesn't matter on single band
    2111             :     }
    2112             : }
    2113             : 
    2114             : // Test GDALDataset::GetRawBinaryLayout() implementations
    2115           4 : TEST_F(test_gdal, GetRawBinaryLayout_PDS4)
    2116             : {
    2117           1 :     if (GDALGetDriverByName("PDS4") == nullptr)
    2118             :     {
    2119           0 :         GTEST_SKIP() << "PDS4 driver missing";
    2120             :     }
    2121             : 
    2122             :     {
    2123             :         GDALDatasetUniquePtr poDS(GDALDataset::Open(
    2124           2 :             GDRIVERS_DATA_DIR "pds4/byte_pds4_cart_1700.xml"));
    2125           1 :         EXPECT_TRUE(poDS != nullptr);
    2126           2 :         GDALDataset::RawBinaryLayout sLayout;
    2127           1 :         EXPECT_TRUE(poDS->GetRawBinaryLayout(sLayout));
    2128           1 :         EXPECT_TRUE(sLayout.osRawFilename.find("byte_pds4_cart_1700.img") !=
    2129             :                     std::string::npos);
    2130           1 :         EXPECT_EQ(static_cast<int>(sLayout.eInterleaving),
    2131             :                   static_cast<int>(
    2132             :                       GDALDataset::RawBinaryLayout::Interleaving::UNKNOWN));
    2133           1 :         EXPECT_EQ(sLayout.eDataType, GDT_Byte);
    2134           1 :         EXPECT_TRUE(!sLayout.bLittleEndianOrder);
    2135           1 :         EXPECT_EQ(sLayout.nImageOffset, 0U);
    2136           1 :         EXPECT_EQ(sLayout.nPixelOffset, 1);
    2137           1 :         EXPECT_EQ(sLayout.nLineOffset, 20);
    2138           1 :         EXPECT_EQ(sLayout.nBandOffset, 0);  // doesn't matter on single band
    2139             :     }
    2140             : }
    2141             : 
    2142             : // Test TileMatrixSet
    2143           4 : TEST_F(test_gdal, TileMatrixSet)
    2144             : {
    2145           1 :     if (getenv("SKIP_TILEMATRIXSET_TEST") != nullptr)
    2146           0 :         GTEST_SKIP() << "Test skipped due to SKIP_TILEMATRIXSET_TEST being set";
    2147             : 
    2148             :     {
    2149           2 :         auto l = gdal::TileMatrixSet::listPredefinedTileMatrixSets();
    2150           1 :         EXPECT_TRUE(l.find("GoogleMapsCompatible") != l.end());
    2151           1 :         EXPECT_TRUE(l.find("NZTM2000") != l.end());
    2152             :     }
    2153             : 
    2154             :     {
    2155           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    2156           1 :         EXPECT_TRUE(gdal::TileMatrixSet::parse("i_dont_exist") == nullptr);
    2157           1 :         CPLPopErrorHandler();
    2158             :     }
    2159             : 
    2160             :     {
    2161           1 :         CPLErrorReset();
    2162           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    2163             :         // Invalid JSON
    2164           1 :         EXPECT_TRUE(gdal::TileMatrixSet::parse(
    2165             :                         "http://127.0.0.1:32767/example.json") == nullptr);
    2166           1 :         CPLPopErrorHandler();
    2167           1 :         EXPECT_TRUE(CPLGetLastErrorType() != 0);
    2168             :     }
    2169             : 
    2170             :     {
    2171           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    2172             :         // Invalid JSON
    2173           1 :         EXPECT_TRUE(gdal::TileMatrixSet::parse(
    2174             :                         "{\"type\": \"TileMatrixSetType\" invalid") == nullptr);
    2175           1 :         CPLPopErrorHandler();
    2176             :     }
    2177             : 
    2178             :     {
    2179           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    2180             :         // No tileMatrix
    2181           1 :         EXPECT_TRUE(gdal::TileMatrixSet::parse(
    2182             :                         "{\"type\": \"TileMatrixSetType\" }") == nullptr);
    2183           1 :         CPLPopErrorHandler();
    2184             :     }
    2185             : 
    2186             :     {
    2187           2 :         auto poTMS = gdal::TileMatrixSet::parse("LINZAntarticaMapTileGrid");
    2188           1 :         EXPECT_TRUE(poTMS != nullptr);
    2189           1 :         if (poTMS)
    2190             :         {
    2191           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTopLeft());
    2192           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTileSize());
    2193           1 :             EXPECT_TRUE(poTMS->hasOnlyPowerOfTwoVaryingScales());
    2194           1 :             EXPECT_TRUE(!poTMS->hasVariableMatrixWidth());
    2195             :         }
    2196             :     }
    2197             : 
    2198             :     {
    2199           2 :         auto poTMS = gdal::TileMatrixSet::parse("NZTM2000");
    2200           1 :         EXPECT_TRUE(poTMS != nullptr);
    2201           1 :         if (poTMS)
    2202             :         {
    2203           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTopLeft());
    2204           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTileSize());
    2205           1 :             EXPECT_TRUE(!poTMS->hasOnlyPowerOfTwoVaryingScales());
    2206           1 :             EXPECT_TRUE(!poTMS->hasVariableMatrixWidth());
    2207             :         }
    2208             :     }
    2209             : 
    2210             :     // Inline JSON with minimal structure
    2211             :     {
    2212             :         auto poTMS = gdal::TileMatrixSet::parse(
    2213             :             "{\"type\": \"TileMatrixSetType\", \"supportedCRS\": "
    2214             :             "\"http://www.opengis.net/def/crs/OGC/1.3/CRS84\", \"tileMatrix\": "
    2215           2 :             "[{ \"topLeftCorner\": [-180, 90],\"scaleDenominator\":1.0}] }");
    2216           1 :         EXPECT_TRUE(poTMS != nullptr);
    2217           1 :         if (poTMS)
    2218             :         {
    2219           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTopLeft());
    2220           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTileSize());
    2221           1 :             EXPECT_TRUE(poTMS->hasOnlyPowerOfTwoVaryingScales());
    2222           1 :             EXPECT_TRUE(!poTMS->hasVariableMatrixWidth());
    2223             :         }
    2224             :     }
    2225             : 
    2226             :     // Invalid scaleDenominator
    2227             :     {
    2228           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    2229           1 :         EXPECT_TRUE(gdal::TileMatrixSet::parse(
    2230             :                         "{\"type\": \"TileMatrixSetType\", \"supportedCRS\": "
    2231             :                         "\"http://www.opengis.net/def/crs/OGC/1.3/CRS84\", "
    2232             :                         "\"tileMatrix\": [{ \"topLeftCorner\": [-180, "
    2233             :                         "90],\"scaleDenominator\":0.0}] }") == nullptr);
    2234           1 :         CPLPopErrorHandler();
    2235             :     }
    2236             : 
    2237             :     {
    2238           1 :         const char *pszJSON = "{"
    2239             :                               "    \"type\": \"TileMatrixSetType\","
    2240             :                               "    \"title\": \"CRS84 for the World\","
    2241             :                               "    \"identifier\": \"WorldCRS84Quad\","
    2242             :                               "    \"abstract\": \"my abstract\","
    2243             :                               "    \"boundingBox\":"
    2244             :                               "    {"
    2245             :                               "        \"type\": \"BoundingBoxType\","
    2246             :                               "        \"crs\": "
    2247             :                               "\"http://www.opengis.net/def/crs/OGC/1.X/"
    2248             :                               "CRS84\","  // 1.3 modified to 1.X to test
    2249             :                                           // difference with supportedCRS
    2250             :                               "        \"lowerCorner\": [-180, -90],"
    2251             :                               "        \"upperCorner\": [180, 90]"
    2252             :                               "    },"
    2253             :                               "    \"supportedCRS\": "
    2254             :                               "\"http://www.opengis.net/def/crs/OGC/1.3/"
    2255             :                               "CRS84\","
    2256             :                               "    \"wellKnownScaleSet\": "
    2257             :                               "\"http://www.opengis.net/def/wkss/OGC/1.0/"
    2258             :                               "GoogleCRS84Quad\","
    2259             :                               "    \"tileMatrix\":"
    2260             :                               "    ["
    2261             :                               "        {"
    2262             :                               "            \"type\": \"TileMatrixType\","
    2263             :                               "            \"identifier\": \"0\","
    2264             :                               "            \"scaleDenominator\": "
    2265             :                               "279541132.014358,"
    2266             :                               "            \"topLeftCorner\": [-180, 90],"
    2267             :                               "            \"tileWidth\": 256,"
    2268             :                               "            \"tileHeight\": 256,"
    2269             :                               "            \"matrixWidth\": 2,"
    2270             :                               "            \"matrixHeight\": 1"
    2271             :                               "        },"
    2272             :                               "        {"
    2273             :                               "            \"type\": \"TileMatrixType\","
    2274             :                               "            \"identifier\": \"1\","
    2275             :                               "            \"scaleDenominator\": "
    2276             :                               "139770566.007179,"
    2277             :                               "            \"topLeftCorner\": [-180, 90],"
    2278             :                               "            \"tileWidth\": 256,"
    2279             :                               "            \"tileHeight\": 256,"
    2280             :                               "            \"matrixWidth\": 4,"
    2281             :                               "            \"matrixHeight\": 2"
    2282             :                               "        }"
    2283             :                               "    ]"
    2284             :                               "}";
    2285           1 :         VSIFCloseL(VSIFileFromMemBuffer(
    2286             :             "/vsimem/tmp.json",
    2287             :             reinterpret_cast<GByte *>(const_cast<char *>(pszJSON)),
    2288           1 :             strlen(pszJSON), false));
    2289           1 :         auto poTMS = gdal::TileMatrixSet::parse("/vsimem/tmp.json");
    2290           1 :         VSIUnlink("/vsimem/tmp.json");
    2291             : 
    2292           1 :         EXPECT_TRUE(poTMS != nullptr);
    2293           1 :         if (poTMS)
    2294             :         {
    2295           1 :             EXPECT_EQ(poTMS->title(), "CRS84 for the World");
    2296           1 :             EXPECT_EQ(poTMS->identifier(), "WorldCRS84Quad");
    2297           1 :             EXPECT_EQ(poTMS->abstract(), "my abstract");
    2298           1 :             EXPECT_EQ(poTMS->crs(),
    2299             :                       "http://www.opengis.net/def/crs/OGC/1.3/CRS84");
    2300           1 :             EXPECT_EQ(
    2301             :                 poTMS->wellKnownScaleSet(),
    2302             :                 "http://www.opengis.net/def/wkss/OGC/1.0/GoogleCRS84Quad");
    2303           1 :             EXPECT_EQ(poTMS->bbox().mCrs,
    2304             :                       "http://www.opengis.net/def/crs/OGC/1.X/CRS84");
    2305           1 :             EXPECT_EQ(poTMS->bbox().mLowerCornerX, -180.0);
    2306           1 :             EXPECT_EQ(poTMS->bbox().mLowerCornerY, -90.0);
    2307           1 :             EXPECT_EQ(poTMS->bbox().mUpperCornerX, 180.0);
    2308           1 :             EXPECT_EQ(poTMS->bbox().mUpperCornerY, 90.0);
    2309           1 :             ASSERT_EQ(poTMS->tileMatrixList().size(), 2U);
    2310           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTopLeft());
    2311           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTileSize());
    2312           1 :             EXPECT_TRUE(poTMS->hasOnlyPowerOfTwoVaryingScales());
    2313           1 :             EXPECT_TRUE(!poTMS->hasVariableMatrixWidth());
    2314           1 :             const auto &tm = poTMS->tileMatrixList()[0];
    2315           1 :             EXPECT_EQ(tm.mId, "0");
    2316           1 :             EXPECT_EQ(tm.mScaleDenominator, 279541132.014358);
    2317           1 :             EXPECT_TRUE(fabs(tm.mResX - tm.mScaleDenominator * 0.28e-3 /
    2318             :                                             (6378137. * M_PI / 180)) < 1e-10);
    2319           1 :             EXPECT_TRUE(fabs(tm.mResX - 180. / 256) < 1e-10);
    2320           1 :             EXPECT_EQ(tm.mResY, tm.mResX);
    2321           1 :             EXPECT_EQ(tm.mTopLeftX, -180.0);
    2322           1 :             EXPECT_EQ(tm.mTopLeftY, 90.0);
    2323           1 :             EXPECT_EQ(tm.mTileWidth, 256);
    2324           1 :             EXPECT_EQ(tm.mTileHeight, 256);
    2325           1 :             EXPECT_EQ(tm.mMatrixWidth, 2);
    2326           1 :             EXPECT_EQ(tm.mMatrixHeight, 1);
    2327             :         }
    2328             :     }
    2329             : 
    2330             :     {
    2331             :         auto poTMS = gdal::TileMatrixSet::parse(
    2332             :             "{"
    2333             :             "    \"type\": \"TileMatrixSetType\","
    2334             :             "    \"title\": \"CRS84 for the World\","
    2335             :             "    \"identifier\": \"WorldCRS84Quad\","
    2336             :             "    \"boundingBox\":"
    2337             :             "    {"
    2338             :             "        \"type\": \"BoundingBoxType\","
    2339             :             "        \"crs\": "
    2340             :             "\"http://www.opengis.net/def/crs/OGC/1.X/"
    2341             :             "CRS84\","  // 1.3 modified to 1.X to test
    2342             :                         // difference with supportedCRS
    2343             :             "        \"lowerCorner\": [-180, -90],"
    2344             :             "        \"upperCorner\": [180, 90]"
    2345             :             "    },"
    2346             :             "    \"supportedCRS\": "
    2347             :             "\"http://www.opengis.net/def/crs/OGC/1.3/"
    2348             :             "CRS84\","
    2349             :             "    \"wellKnownScaleSet\": "
    2350             :             "\"http://www.opengis.net/def/wkss/OGC/1.0/"
    2351             :             "GoogleCRS84Quad\","
    2352             :             "    \"tileMatrix\":"
    2353             :             "    ["
    2354             :             "        {"
    2355             :             "            \"type\": \"TileMatrixType\","
    2356             :             "            \"identifier\": \"0\","
    2357             :             "            \"scaleDenominator\": "
    2358             :             "279541132.014358,"
    2359             :             "            \"topLeftCorner\": [-180, 90],"
    2360             :             "            \"tileWidth\": 256,"
    2361             :             "            \"tileHeight\": 256,"
    2362             :             "            \"matrixWidth\": 2,"
    2363             :             "            \"matrixHeight\": 1"
    2364             :             "        },"
    2365             :             "        {"
    2366             :             "            \"type\": \"TileMatrixType\","
    2367             :             "            \"identifier\": \"1\","
    2368             :             "            \"scaleDenominator\": 100000000,"
    2369             :             "            \"topLeftCorner\": [-123, 90],"
    2370             :             "            \"tileWidth\": 128,"
    2371             :             "            \"tileHeight\": 256,"
    2372             :             "            \"matrixWidth\": 4,"
    2373             :             "            \"matrixHeight\": 2,"
    2374             :             "            \"variableMatrixWidth\": [{"
    2375             :             "               \"type\": "
    2376             :             "\"VariableMatrixWidthType\","
    2377             :             "               \"coalesce\" : 2,"
    2378             :             "               \"minTileRow\": 0,"
    2379             :             "               \"maxTileRow\": 1"
    2380             :             "            }]"
    2381             :             "        }"
    2382             :             "    ]"
    2383           1 :             "}");
    2384           1 :         EXPECT_TRUE(poTMS != nullptr);
    2385           1 :         if (poTMS)
    2386             :         {
    2387           1 :             ASSERT_EQ(poTMS->tileMatrixList().size(), 2U);
    2388           1 :             EXPECT_TRUE(!poTMS->haveAllLevelsSameTopLeft());
    2389           1 :             EXPECT_TRUE(!poTMS->haveAllLevelsSameTileSize());
    2390           1 :             EXPECT_TRUE(!poTMS->hasOnlyPowerOfTwoVaryingScales());
    2391           1 :             EXPECT_TRUE(poTMS->hasVariableMatrixWidth());
    2392           1 :             const auto &tm = poTMS->tileMatrixList()[1];
    2393           1 :             EXPECT_EQ(tm.mVariableMatrixWidthList.size(), 1U);
    2394           1 :             const auto &vmw = tm.mVariableMatrixWidthList[0];
    2395           1 :             EXPECT_EQ(vmw.mCoalesce, 2);
    2396           1 :             EXPECT_EQ(vmw.mMinTileRow, 0);
    2397           1 :             EXPECT_EQ(vmw.mMaxTileRow, 1);
    2398             :         }
    2399             :     }
    2400             : 
    2401             :     {
    2402             :         auto poTMS = gdal::TileMatrixSet::parse(
    2403             :             "{"
    2404             :             "    \"identifier\" : \"CDBGlobalGrid\","
    2405             :             "    \"title\" : \"CDBGlobalGrid\","
    2406             :             "    \"boundingBox\" : {"
    2407             :             "        \"crs\" : \"http://www.opengis.net/def/crs/EPSG/0/4326\","
    2408             :             "        \"lowerCorner\" : ["
    2409             :             "            -90,"
    2410             :             "            -180"
    2411             :             "        ],"
    2412             :             "        \"upperCorner\" : ["
    2413             :             "            90,"
    2414             :             "            180"
    2415             :             "        ]"
    2416             :             "    },"
    2417             :             "    \"supportedCRS\" : "
    2418             :             "\"http://www.opengis.net/def/crs/EPSG/0/4326\","
    2419             :             "    \"wellKnownScaleSet\" : "
    2420             :             "\"http://www.opengis.net/def/wkss/OGC/1.0/CDBGlobalGrid\","
    2421             :             "    \"tileMatrix\" : ["
    2422             :             "        {"
    2423             :             "            \"identifier\" : \"-10\","
    2424             :             "            \"scaleDenominator\" : 397569609.975977063179,"
    2425             :             "            \"matrixWidth\" : 360,"
    2426             :             "            \"matrixHeight\" : 180,"
    2427             :             "            \"tileWidth\" : 1,"
    2428             :             "            \"tileHeight\" : 1,"
    2429             :             "            \"topLeftCorner\" : ["
    2430             :             "                90,"
    2431             :             "                -180"
    2432             :             "            ],"
    2433             :             "            \"variableMatrixWidth\" : ["
    2434             :             "                {"
    2435             :             "                \"coalesce\" : 12,"
    2436             :             "                \"minTileRow\" : 0,"
    2437             :             "                \"maxTileRow\" : 0"
    2438             :             "                },"
    2439             :             "                {"
    2440             :             "                \"coalesce\" : 12,"
    2441             :             "                \"minTileRow\" : 179,"
    2442             :             "                \"maxTileRow\" : 179"
    2443             :             "                }"
    2444             :             "            ]"
    2445             :             "        }"
    2446             :             "    ]"
    2447           1 :             "}");
    2448           1 :         EXPECT_TRUE(poTMS != nullptr);
    2449           1 :         if (poTMS)
    2450             :         {
    2451           1 :             ASSERT_EQ(poTMS->tileMatrixList().size(), 1U);
    2452           1 :             const auto &tm = poTMS->tileMatrixList()[0];
    2453           1 :             EXPECT_EQ(tm.mVariableMatrixWidthList.size(), 2U);
    2454           1 :             const auto &vmw = tm.mVariableMatrixWidthList[0];
    2455           1 :             EXPECT_EQ(vmw.mCoalesce, 12);
    2456           1 :             EXPECT_EQ(vmw.mMinTileRow, 0);
    2457           1 :             EXPECT_EQ(vmw.mMaxTileRow, 0);
    2458             :         }
    2459             :     }
    2460             : 
    2461             :     // TMS v2 (truncated version of https://maps.gnosis.earth/ogcapi/tileMatrixSets/GNOSISGlobalGrid?f=json)
    2462             :     {
    2463             :         auto poTMS = gdal::TileMatrixSet::parse(
    2464             :             "{"
    2465             :             "   \"id\" : \"GNOSISGlobalGrid\","
    2466             :             "   \"title\" : \"GNOSISGlobalGrid\","
    2467             :             "   \"uri\" : "
    2468             :             "\"http://www.opengis.net/def/tilematrixset/OGC/1.0/"
    2469             :             "GNOSISGlobalGrid\","
    2470             :             "   \"description\": \"added for testing\","
    2471             :             "   \"crs\" : \"http://www.opengis.net/def/crs/EPSG/0/4326\","
    2472             :             "   \"orderedAxes\" : ["
    2473             :             "      \"Lat\","
    2474             :             "      \"Lon\""
    2475             :             "   ],"
    2476             :             "   \"wellKnownScaleSet\" : "
    2477             :             "\"http://www.opengis.net/def/wkss/OGC/1.0/GoogleCRS84Quad\","
    2478             :             "   \"tileMatrices\" : ["
    2479             :             "      {"
    2480             :             "         \"id\" : \"0\","
    2481             :             "         \"scaleDenominator\" : 139770566.0071794390678,"
    2482             :             "         \"cellSize\" : 0.3515625,"
    2483             :             "         \"cornerOfOrigin\" : \"topLeft\","
    2484             :             "         \"pointOfOrigin\" : [ 90, -180 ],"
    2485             :             "         \"matrixWidth\" : 4,"
    2486             :             "         \"matrixHeight\" : 2,"
    2487             :             "         \"tileWidth\" : 256,"
    2488             :             "         \"tileHeight\" : 256"
    2489             :             "      },"
    2490             :             "      {"
    2491             :             "         \"id\" : \"1\","
    2492             :             "         \"scaleDenominator\" : 69885283.0035897195339,"
    2493             :             "         \"cellSize\" : 0.17578125,"
    2494             :             "         \"cornerOfOrigin\" : \"topLeft\","
    2495             :             "         \"pointOfOrigin\" : [ 90, -180 ],"
    2496             :             "         \"matrixWidth\" : 8,"
    2497             :             "         \"matrixHeight\" : 4,"
    2498             :             "         \"tileWidth\" : 256,"
    2499             :             "         \"tileHeight\" : 256,"
    2500             :             "         \"variableMatrixWidths\" : ["
    2501             :             "            { \"coalesce\" : 2, \"minTileRow\" : 0, "
    2502             :             "\"maxTileRow\" : 0 },"
    2503             :             "            { \"coalesce\" : 2, \"minTileRow\" : 3, "
    2504             :             "\"maxTileRow\" : 3 }"
    2505             :             "         ]"
    2506             :             "      }"
    2507             :             "   ]"
    2508           1 :             "}");
    2509           1 :         EXPECT_TRUE(poTMS != nullptr);
    2510           1 :         if (poTMS)
    2511             :         {
    2512           1 :             EXPECT_EQ(poTMS->title(), "GNOSISGlobalGrid");
    2513           1 :             EXPECT_EQ(poTMS->identifier(), "GNOSISGlobalGrid");
    2514           1 :             EXPECT_EQ(poTMS->abstract(), "added for testing");
    2515           1 :             EXPECT_EQ(poTMS->crs(),
    2516             :                       "http://www.opengis.net/def/crs/EPSG/0/4326");
    2517           1 :             EXPECT_EQ(
    2518             :                 poTMS->wellKnownScaleSet(),
    2519             :                 "http://www.opengis.net/def/wkss/OGC/1.0/GoogleCRS84Quad");
    2520           1 :             ASSERT_EQ(poTMS->tileMatrixList().size(), 2U);
    2521           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTopLeft());
    2522           1 :             EXPECT_TRUE(poTMS->haveAllLevelsSameTileSize());
    2523           1 :             EXPECT_TRUE(poTMS->hasOnlyPowerOfTwoVaryingScales());
    2524             :             {
    2525           1 :                 const auto &tm = poTMS->tileMatrixList()[0];
    2526           1 :                 EXPECT_EQ(tm.mId, "0");
    2527           1 :                 EXPECT_EQ(tm.mScaleDenominator, 139770566.0071794390678);
    2528           1 :                 EXPECT_TRUE(fabs(tm.mResX - tm.mScaleDenominator * 0.28e-3 /
    2529             :                                                 (6378137. * M_PI / 180)) <
    2530             :                             1e-10);
    2531           1 :                 EXPECT_EQ(tm.mResY, tm.mResX);
    2532           1 :                 EXPECT_EQ(tm.mTopLeftX, 90.0);
    2533           1 :                 EXPECT_EQ(tm.mTopLeftY, -180.0);
    2534           1 :                 EXPECT_EQ(tm.mTileWidth, 256);
    2535           1 :                 EXPECT_EQ(tm.mTileHeight, 256);
    2536           1 :                 EXPECT_EQ(tm.mMatrixWidth, 4);
    2537           1 :                 EXPECT_EQ(tm.mMatrixHeight, 2);
    2538             :             }
    2539             : 
    2540           1 :             EXPECT_TRUE(poTMS->hasVariableMatrixWidth());
    2541             :             {
    2542           1 :                 const auto &tm = poTMS->tileMatrixList()[1];
    2543           1 :                 EXPECT_EQ(tm.mVariableMatrixWidthList.size(), 2U);
    2544           1 :                 const auto &vmw = tm.mVariableMatrixWidthList[1];
    2545           1 :                 EXPECT_EQ(vmw.mCoalesce, 2);
    2546           1 :                 EXPECT_EQ(vmw.mMinTileRow, 3);
    2547           1 :                 EXPECT_EQ(vmw.mMaxTileRow, 3);
    2548             :             }
    2549             :         }
    2550             :     }
    2551             : 
    2552             :     // TMS v2 with crs.uri
    2553             :     {
    2554             :         auto poTMS = gdal::TileMatrixSet::parse(
    2555             :             "{"
    2556             :             "   \"id\" : \"test\","
    2557             :             "   \"title\" : \"test\","
    2558             :             "   \"uri\" : "
    2559             :             "\"http://www.opengis.net/def/tilematrixset/OGC/1.0/test\","
    2560             :             "   \"crs\" : {\"uri\": "
    2561             :             "\"http://www.opengis.net/def/crs/EPSG/0/4326\"},"
    2562             :             "   \"orderedAxes\" : ["
    2563             :             "      \"Lat\","
    2564             :             "      \"Lon\""
    2565             :             "   ],"
    2566             :             "   \"wellKnownScaleSet\" : "
    2567             :             "\"http://www.opengis.net/def/wkss/OGC/1.0/GoogleCRS84Quad\","
    2568             :             "   \"tileMatrices\" : ["
    2569             :             "      {"
    2570             :             "         \"id\" : \"0\","
    2571             :             "         \"scaleDenominator\" : 139770566.0071794390678,"
    2572             :             "         \"cellSize\" : 0.3515625,"
    2573             :             "         \"cornerOfOrigin\" : \"topLeft\","
    2574             :             "         \"pointOfOrigin\" : [ 90, -180 ],"
    2575             :             "         \"matrixWidth\" : 4,"
    2576             :             "         \"matrixHeight\" : 2,"
    2577             :             "         \"tileWidth\" : 256,"
    2578             :             "         \"tileHeight\" : 256"
    2579             :             "      }"
    2580             :             "   ]"
    2581           2 :             "}");
    2582           1 :         EXPECT_TRUE(poTMS != nullptr);
    2583           1 :         if (poTMS)
    2584             :         {
    2585           1 :             EXPECT_EQ(poTMS->crs(),
    2586             :                       "http://www.opengis.net/def/crs/EPSG/0/4326");
    2587             :         }
    2588             :     }
    2589             : 
    2590             :     // TMS v2 with crs.wkt
    2591             :     {
    2592             :         auto poTMS = gdal::TileMatrixSet::parse(
    2593             :             "{"
    2594             :             "   \"id\" : \"test\","
    2595             :             "   \"title\" : \"test\","
    2596             :             "   \"uri\" : "
    2597             :             "\"http://www.opengis.net/def/tilematrixset/OGC/1.0/test\","
    2598             :             "   \"crs\" : {\"wkt\": \"GEOGCRS[\\\"WGS 84\\\","
    2599             :             "ENSEMBLE[\\\"World Geodetic System 1984 ensemble\\\","
    2600             :             "MEMBER[\\\"World Geodetic System 1984 (Transit)\\\"],"
    2601             :             "MEMBER[\\\"World Geodetic System 1984 (G730)\\\"],"
    2602             :             "MEMBER[\\\"World Geodetic System 1984 (G873)\\\"],"
    2603             :             "MEMBER[\\\"World Geodetic System 1984 (G1150)\\\"],"
    2604             :             "MEMBER[\\\"World Geodetic System 1984 (G1674)\\\"],"
    2605             :             "MEMBER[\\\"World Geodetic System 1984 (G1762)\\\"],"
    2606             :             "MEMBER[\\\"World Geodetic System 1984 (G2139)\\\"],"
    2607             :             "MEMBER[\\\"World Geodetic System 1984 (G2296)\\\"],"
    2608             :             "ELLIPSOID[\\\"WGS 84\\\",6378137,298.257223563,"
    2609             :             "LENGTHUNIT[\\\"metre\\\",1]],"
    2610             :             "ENSEMBLEACCURACY[2.0]],"
    2611             :             "PRIMEM[\\\"Greenwich\\\",0,"
    2612             :             "ANGLEUNIT[\\\"degree\\\",0.0174532925199433]],"
    2613             :             "CS[ellipsoidal,2],"
    2614             :             "AXIS[\\\"geodetic latitude (Lat)\\\",north,"
    2615             :             "ORDER[1],"
    2616             :             "ANGLEUNIT[\\\"degree\\\",0.0174532925199433]],"
    2617             :             "AXIS[\\\"geodetic longitude (Lon)\\\",east,"
    2618             :             "ORDER[2],"
    2619             :             "ANGLEUNIT[\\\"degree\\\",0.0174532925199433]],"
    2620             :             "USAGE["
    2621             :             "SCOPE[\\\"Horizontal component of 3D system.\\\"],"
    2622             :             "AREA[\\\"World.\\\"],"
    2623             :             "BBOX[-90,-180,90,180]],"
    2624             :             "ID[\\\"EPSG\\\",4326]]\" },"
    2625             :             "   \"orderedAxes\" : ["
    2626             :             "      \"Lat\","
    2627             :             "      \"Lon\""
    2628             :             "   ],"
    2629             :             "   \"wellKnownScaleSet\" : "
    2630             :             "\"http://www.opengis.net/def/wkss/OGC/1.0/GoogleCRS84Quad\","
    2631             :             "   \"tileMatrices\" : ["
    2632             :             "      {"
    2633             :             "         \"id\" : \"0\","
    2634             :             "         \"scaleDenominator\" : 139770566.0071794390678,"
    2635             :             "         \"cellSize\" : 0.3515625,"
    2636             :             "         \"cornerOfOrigin\" : \"topLeft\","
    2637             :             "         \"pointOfOrigin\" : [ 90, -180 ],"
    2638             :             "         \"matrixWidth\" : 4,"
    2639             :             "         \"matrixHeight\" : 2,"
    2640             :             "         \"tileWidth\" : 256,"
    2641             :             "         \"tileHeight\" : 256"
    2642             :             "      }"
    2643             :             "   ]"
    2644           2 :             "}");
    2645           1 :         EXPECT_TRUE(poTMS != nullptr);
    2646           1 :         if (poTMS)
    2647             :         {
    2648           1 :             EXPECT_TRUE(
    2649             :                 STARTS_WITH(poTMS->crs().c_str(), "GEOGCRS[\"WGS 84\""));
    2650             :         }
    2651             :     }
    2652             : 
    2653             :     // TMS v2 with crs.wkt with JSON content
    2654             :     {
    2655             :         auto poTMS = gdal::TileMatrixSet::parse(
    2656             :             "{"
    2657             :             "   \"id\" : \"test\","
    2658             :             "   \"title\" : \"test\","
    2659             :             "   \"uri\" : "
    2660             :             "\"http://www.opengis.net/def/tilematrixset/OGC/1.0/test\","
    2661             :             "   \"crs\" : {\"wkt\": "
    2662             :             "{"
    2663             :             "  \"type\": \"GeographicCRS\","
    2664             :             "  \"name\": \"WGS 84\","
    2665             :             "  \"datum_ensemble\": {"
    2666             :             "    \"name\": \"World Geodetic System 1984 ensemble\","
    2667             :             "    \"members\": ["
    2668             :             "      {"
    2669             :             "        \"name\": \"World Geodetic System 1984 (Transit)\","
    2670             :             "        \"id\": {"
    2671             :             "          \"authority\": \"EPSG\","
    2672             :             "          \"code\": 1166"
    2673             :             "        }"
    2674             :             "      },"
    2675             :             "      {"
    2676             :             "        \"name\": \"World Geodetic System 1984 (G730)\","
    2677             :             "        \"id\": {"
    2678             :             "          \"authority\": \"EPSG\","
    2679             :             "          \"code\": 1152"
    2680             :             "        }"
    2681             :             "      },"
    2682             :             "      {"
    2683             :             "        \"name\": \"World Geodetic System 1984 (G873)\","
    2684             :             "        \"id\": {"
    2685             :             "          \"authority\": \"EPSG\","
    2686             :             "          \"code\": 1153"
    2687             :             "        }"
    2688             :             "      },"
    2689             :             "      {"
    2690             :             "        \"name\": \"World Geodetic System 1984 (G1150)\","
    2691             :             "        \"id\": {"
    2692             :             "          \"authority\": \"EPSG\","
    2693             :             "          \"code\": 1154"
    2694             :             "        }"
    2695             :             "      },"
    2696             :             "      {"
    2697             :             "        \"name\": \"World Geodetic System 1984 (G1674)\","
    2698             :             "        \"id\": {"
    2699             :             "          \"authority\": \"EPSG\","
    2700             :             "          \"code\": 1155"
    2701             :             "        }"
    2702             :             "      },"
    2703             :             "      {"
    2704             :             "        \"name\": \"World Geodetic System 1984 (G1762)\","
    2705             :             "        \"id\": {"
    2706             :             "          \"authority\": \"EPSG\","
    2707             :             "          \"code\": 1156"
    2708             :             "        }"
    2709             :             "      },"
    2710             :             "      {"
    2711             :             "        \"name\": \"World Geodetic System 1984 (G2139)\","
    2712             :             "        \"id\": {"
    2713             :             "          \"authority\": \"EPSG\","
    2714             :             "          \"code\": 1309"
    2715             :             "        }"
    2716             :             "      },"
    2717             :             "      {"
    2718             :             "        \"name\": \"World Geodetic System 1984 (G2296)\","
    2719             :             "        \"id\": {"
    2720             :             "          \"authority\": \"EPSG\","
    2721             :             "          \"code\": 1383"
    2722             :             "        }"
    2723             :             "      }"
    2724             :             "    ],"
    2725             :             "    \"ellipsoid\": {"
    2726             :             "      \"name\": \"WGS 84\","
    2727             :             "      \"semi_major_axis\": 6378137,"
    2728             :             "      \"inverse_flattening\": 298.257223563"
    2729             :             "    },"
    2730             :             "    \"accuracy\": \"2.0\","
    2731             :             "    \"id\": {"
    2732             :             "      \"authority\": \"EPSG\","
    2733             :             "      \"code\": 6326"
    2734             :             "    }"
    2735             :             "  },"
    2736             :             "  \"coordinate_system\": {"
    2737             :             "    \"subtype\": \"ellipsoidal\","
    2738             :             "    \"axis\": ["
    2739             :             "      {"
    2740             :             "        \"name\": \"Geodetic latitude\","
    2741             :             "        \"abbreviation\": \"Lat\","
    2742             :             "        \"direction\": \"north\","
    2743             :             "        \"unit\": \"degree\""
    2744             :             "      },"
    2745             :             "      {"
    2746             :             "        \"name\": \"Geodetic longitude\","
    2747             :             "        \"abbreviation\": \"Lon\","
    2748             :             "        \"direction\": \"east\","
    2749             :             "        \"unit\": \"degree\""
    2750             :             "      }"
    2751             :             "    ]"
    2752             :             "  },"
    2753             :             "  \"scope\": \"Horizontal component of 3D system.\","
    2754             :             "  \"area\": \"World.\","
    2755             :             "  \"bbox\": {"
    2756             :             "    \"south_latitude\": -90,"
    2757             :             "    \"west_longitude\": -180,"
    2758             :             "    \"north_latitude\": 90,"
    2759             :             "    \"east_longitude\": 180"
    2760             :             "  },"
    2761             :             "  \"id\": {"
    2762             :             "    \"authority\": \"EPSG\","
    2763             :             "    \"code\": 4326"
    2764             :             "  }"
    2765             :             "}"
    2766             :             "},"
    2767             :             "   \"orderedAxes\" : ["
    2768             :             "      \"Lat\","
    2769             :             "      \"Lon\""
    2770             :             "   ],"
    2771             :             "   \"wellKnownScaleSet\" : "
    2772             :             "\"http://www.opengis.net/def/wkss/OGC/1.0/GoogleCRS84Quad\","
    2773             :             "   \"tileMatrices\" : ["
    2774             :             "      {"
    2775             :             "         \"id\" : \"0\","
    2776             :             "         \"scaleDenominator\" : 139770566.0071794390678,"
    2777             :             "         \"cellSize\" : 0.3515625,"
    2778             :             "         \"cornerOfOrigin\" : \"topLeft\","
    2779             :             "         \"pointOfOrigin\" : [ 90, -180 ],"
    2780             :             "         \"matrixWidth\" : 4,"
    2781             :             "         \"matrixHeight\" : 2,"
    2782             :             "         \"tileWidth\" : 256,"
    2783             :             "         \"tileHeight\" : 256"
    2784             :             "      }"
    2785             :             "   ]"
    2786           2 :             "}");
    2787           1 :         EXPECT_TRUE(poTMS != nullptr);
    2788           1 :         if (poTMS)
    2789             :         {
    2790           1 :             EXPECT_TRUE(STARTS_WITH(poTMS->crs().c_str(),
    2791             :                                     "{ \"type\": \"GeographicCRS\""));
    2792             :         }
    2793             :     }
    2794             : }
    2795             : 
    2796             : // Test that PCIDSK GetMetadataItem() return is stable
    2797           4 : TEST_F(test_gdal, PCIDSK_GetMetadataItem)
    2798             : {
    2799           1 :     auto poDrv = GDALDriver::FromHandle(GDALGetDriverByName("PCIDSK"));
    2800           1 :     if (poDrv == nullptr)
    2801           0 :         GTEST_SKIP() << "PCIDSK driver missing";
    2802             : 
    2803             :     GDALDatasetUniquePtr poDS(
    2804           2 :         poDrv->Create("/vsimem/tmp.pix", 1, 1, 1, GDT_Byte, nullptr));
    2805           1 :     EXPECT_TRUE(poDS != nullptr);
    2806           1 :     poDS->SetMetadataItem("FOO", "BAR");
    2807           1 :     poDS->SetMetadataItem("BAR", "BAZ");
    2808           1 :     poDS->GetRasterBand(1)->SetMetadataItem("FOO", "BAR");
    2809           1 :     poDS->GetRasterBand(1)->SetMetadataItem("BAR", "BAZ");
    2810             : 
    2811             :     {
    2812           1 :         const char *psz1 = poDS->GetMetadataItem("FOO");
    2813           1 :         const char *psz2 = poDS->GetMetadataItem("BAR");
    2814           1 :         const char *pszNull = poDS->GetMetadataItem("I_DONT_EXIST");
    2815           1 :         const char *psz3 = poDS->GetMetadataItem("FOO");
    2816           1 :         const char *pszNull2 = poDS->GetMetadataItem("I_DONT_EXIST");
    2817           1 :         const char *psz4 = poDS->GetMetadataItem("BAR");
    2818           1 :         EXPECT_TRUE(psz1 != nullptr);
    2819           1 :         EXPECT_TRUE(psz2 != nullptr);
    2820           1 :         EXPECT_TRUE(psz3 != nullptr);
    2821           1 :         EXPECT_TRUE(psz4 != nullptr);
    2822           1 :         EXPECT_TRUE(pszNull == nullptr);
    2823           1 :         EXPECT_TRUE(pszNull2 == nullptr);
    2824           1 :         EXPECT_EQ(psz1, psz3);
    2825           1 :         EXPECT_TRUE(psz1 != psz2);
    2826           1 :         EXPECT_EQ(psz2, psz4);
    2827           1 :         EXPECT_STREQ(psz1, "BAR");
    2828           1 :         EXPECT_STREQ(psz2, "BAZ");
    2829             :     }
    2830             : 
    2831             :     {
    2832           1 :         auto poBand = poDS->GetRasterBand(1);
    2833           1 :         const char *psz1 = poBand->GetMetadataItem("FOO");
    2834           1 :         const char *psz2 = poBand->GetMetadataItem("BAR");
    2835           1 :         const char *pszNull = poBand->GetMetadataItem("I_DONT_EXIST");
    2836           1 :         const char *psz3 = poBand->GetMetadataItem("FOO");
    2837           1 :         const char *pszNull2 = poBand->GetMetadataItem("I_DONT_EXIST");
    2838           1 :         const char *psz4 = poBand->GetMetadataItem("BAR");
    2839           1 :         EXPECT_TRUE(psz1 != nullptr);
    2840           1 :         EXPECT_TRUE(psz2 != nullptr);
    2841           1 :         EXPECT_TRUE(psz3 != nullptr);
    2842           1 :         EXPECT_TRUE(psz4 != nullptr);
    2843           1 :         EXPECT_TRUE(pszNull == nullptr);
    2844           1 :         EXPECT_TRUE(pszNull2 == nullptr);
    2845           1 :         EXPECT_EQ(psz1, psz3);
    2846           1 :         EXPECT_TRUE(psz1 != psz2);
    2847           1 :         EXPECT_EQ(psz2, psz4);
    2848           1 :         EXPECT_STREQ(psz1, "BAR");
    2849           1 :         EXPECT_STREQ(psz2, "BAZ");
    2850             :     }
    2851             : 
    2852           1 :     poDS.reset();
    2853           1 :     VSIUnlink("/vsimem/tmp.pix");
    2854             : }
    2855             : 
    2856             : // Test GDALBufferHasOnlyNoData()
    2857           4 : TEST_F(test_gdal, GDALBufferHasOnlyNoData)
    2858             : {
    2859             :     /* bool CPL_DLL GDALBufferHasOnlyNoData(const void* pBuffer,
    2860             :                                  double dfNoDataValue,
    2861             :                                  size_t nWidth, size_t nHeight,
    2862             :                                  size_t nLineStride,
    2863             :                                  size_t nComponents,
    2864             :                                  int nBitsPerSample,
    2865             :                                  GDALBufferSampleFormat nSampleFormat);
    2866             :      */
    2867           1 :     EXPECT_TRUE(
    2868             :         GDALBufferHasOnlyNoData("\x00", 0.0, 1, 1, 1, 1, 8, GSF_UNSIGNED_INT));
    2869           1 :     EXPECT_TRUE(
    2870             :         !GDALBufferHasOnlyNoData("\x01", 0.0, 1, 1, 1, 1, 8, GSF_UNSIGNED_INT));
    2871           1 :     EXPECT_TRUE(
    2872             :         GDALBufferHasOnlyNoData("\x00", 0.0, 1, 1, 1, 1, 1, GSF_UNSIGNED_INT));
    2873           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData("\x00\x00", 0.0, 1, 1, 1, 1, 16,
    2874             :                                         GSF_UNSIGNED_INT));
    2875           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData("\x00\x01", 0.0, 1, 1, 1, 1, 16,
    2876             :                                          GSF_UNSIGNED_INT));
    2877           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData("\x00\x01", 0.0, 1, 2, 2, 1, 8,
    2878             :                                         GSF_UNSIGNED_INT));
    2879           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(
    2880             :         "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 0.0, 14, 1,
    2881             :         14, 1, 8, GSF_UNSIGNED_INT));
    2882           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(
    2883             :         "\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 0.0, 14, 1,
    2884             :         14, 1, 8, GSF_UNSIGNED_INT));
    2885           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(
    2886             :         "\x00\x00\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\x00", 0.0, 14, 1,
    2887             :         14, 1, 8, GSF_UNSIGNED_INT));
    2888           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(
    2889             :         "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01", 0.0, 14, 1,
    2890             :         14, 1, 8, GSF_UNSIGNED_INT));
    2891             : 
    2892           1 :     uint8_t uint8val = 1;
    2893           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&uint8val, 1.0, 1, 1, 1, 1, 8,
    2894             :                                         GSF_UNSIGNED_INT));
    2895           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&uint8val, 0.0, 1, 1, 1, 1, 8,
    2896             :                                          GSF_UNSIGNED_INT));
    2897           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&uint8val, 128 + 1, 1, 1, 1, 1, 8,
    2898             :                                          GSF_UNSIGNED_INT));
    2899             : 
    2900           1 :     int8_t int8val = -1;
    2901           1 :     EXPECT_TRUE(
    2902             :         GDALBufferHasOnlyNoData(&int8val, -1.0, 1, 1, 1, 1, 8, GSF_SIGNED_INT));
    2903           1 :     EXPECT_TRUE(
    2904             :         !GDALBufferHasOnlyNoData(&int8val, 0.0, 1, 1, 1, 1, 8, GSF_SIGNED_INT));
    2905           1 :     EXPECT_TRUE(
    2906             :         !GDALBufferHasOnlyNoData(&int8val, 256, 1, 1, 1, 1, 8, GSF_SIGNED_INT));
    2907             : 
    2908           1 :     uint16_t uint16val = 1;
    2909           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&uint16val, 1.0, 1, 1, 1, 1, 16,
    2910             :                                         GSF_UNSIGNED_INT));
    2911           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&uint16val, 0.0, 1, 1, 1, 1, 16,
    2912             :                                          GSF_UNSIGNED_INT));
    2913           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&uint16val, 65536 + 1, 1, 1, 1, 1, 16,
    2914             :                                          GSF_UNSIGNED_INT));
    2915             : 
    2916           1 :     int16_t int16val = -1;
    2917           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&int16val, -1.0, 1, 1, 1, 1, 16,
    2918             :                                         GSF_SIGNED_INT));
    2919           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&int16val, 0.0, 1, 1, 1, 1, 16,
    2920             :                                          GSF_SIGNED_INT));
    2921           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&int16val, 32768, 1, 1, 1, 1, 16,
    2922             :                                          GSF_SIGNED_INT));
    2923             : 
    2924           1 :     uint32_t uint32val = 1;
    2925           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&uint32val, 1.0, 1, 1, 1, 1, 32,
    2926             :                                         GSF_UNSIGNED_INT));
    2927           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&uint32val, 0.0, 1, 1, 1, 1, 32,
    2928             :                                          GSF_UNSIGNED_INT));
    2929           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&uint32val,
    2930             :                                          static_cast<double>(0x100000000LL + 1),
    2931             :                                          1, 1, 1, 1, 32, GSF_UNSIGNED_INT));
    2932             : 
    2933           1 :     int32_t int32val = -1;
    2934           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&int32val, -1.0, 1, 1, 1, 1, 32,
    2935             :                                         GSF_SIGNED_INT));
    2936           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&int32val, 0.0, 1, 1, 1, 1, 32,
    2937             :                                          GSF_SIGNED_INT));
    2938           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&int32val, 0x80000000, 1, 1, 1, 1, 32,
    2939             :                                          GSF_SIGNED_INT));
    2940             : 
    2941           1 :     float float32val = -1;
    2942           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&float32val, -1.0, 1, 1, 1, 1, 32,
    2943             :                                         GSF_FLOATING_POINT));
    2944           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&float32val, 0.0, 1, 1, 1, 1, 32,
    2945             :                                          GSF_FLOATING_POINT));
    2946           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&float32val, 1e50, 1, 1, 1, 1, 32,
    2947             :                                          GSF_FLOATING_POINT));
    2948             : 
    2949           1 :     float float32nan = std::numeric_limits<float>::quiet_NaN();
    2950           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&float32nan, float32nan, 1, 1, 1, 1, 32,
    2951             :                                         GSF_FLOATING_POINT));
    2952           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&float32nan, 0.0, 1, 1, 1, 1, 32,
    2953             :                                          GSF_FLOATING_POINT));
    2954             : 
    2955           1 :     double float64val = -1;
    2956           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&float64val, -1.0, 1, 1, 1, 1, 64,
    2957             :                                         GSF_FLOATING_POINT));
    2958           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&float64val, 0.0, 1, 1, 1, 1, 64,
    2959             :                                          GSF_FLOATING_POINT));
    2960             : 
    2961           1 :     double float64nan = std::numeric_limits<double>::quiet_NaN();
    2962           1 :     EXPECT_TRUE(GDALBufferHasOnlyNoData(&float64nan, float64nan, 1, 1, 1, 1, 64,
    2963             :                                         GSF_FLOATING_POINT));
    2964           1 :     EXPECT_TRUE(!GDALBufferHasOnlyNoData(&float64nan, 0.0, 1, 1, 1, 1, 64,
    2965             :                                          GSF_FLOATING_POINT));
    2966           1 : }
    2967             : 
    2968             : // Test GetRasterNoDataReplacementValue()
    2969           4 : TEST_F(test_gdal, GetRasterNoDataReplacementValue)
    2970             : {
    2971             :     // Test GDT_Byte
    2972           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    2973             :                   GDT_Byte, std::numeric_limits<double>::lowest()),
    2974             :               0);
    2975           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Byte,
    2976             :                                             std::numeric_limits<double>::max()),
    2977             :               0);
    2978           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    2979             :                   GDT_Byte, std::numeric_limits<uint8_t>::lowest()),
    2980             :               std::numeric_limits<uint8_t>::lowest() + 1);
    2981           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    2982             :                   GDT_Byte, std::numeric_limits<uint8_t>::max()),
    2983             :               std::numeric_limits<uint8_t>::max() - 1);
    2984             : 
    2985             :     // Test GDT_Int8
    2986           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    2987             :                   GDT_Int8, std::numeric_limits<double>::lowest()),
    2988             :               0);
    2989           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Int8,
    2990             :                                             std::numeric_limits<double>::max()),
    2991             :               0);
    2992           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    2993             :                   GDT_Int8, std::numeric_limits<int8_t>::lowest()),
    2994             :               std::numeric_limits<int8_t>::lowest() + 1);
    2995           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Int8,
    2996             :                                             std::numeric_limits<int8_t>::max()),
    2997             :               std::numeric_limits<int8_t>::max() - 1);
    2998             : 
    2999             :     // Test GDT_UInt16
    3000           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3001             :                   GDT_UInt16, std::numeric_limits<double>::lowest()),
    3002             :               0);
    3003           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_UInt16,
    3004             :                                             std::numeric_limits<double>::max()),
    3005             :               0);
    3006           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3007             :                   GDT_UInt16, std::numeric_limits<uint16_t>::lowest()),
    3008             :               std::numeric_limits<uint16_t>::lowest() + 1);
    3009           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3010             :                   GDT_UInt16, std::numeric_limits<uint16_t>::max()),
    3011             :               std::numeric_limits<uint16_t>::max() - 1);
    3012             : 
    3013             :     // Test GDT_Int16
    3014           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3015             :                   GDT_Int16, std::numeric_limits<double>::lowest()),
    3016             :               0);
    3017           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Int16,
    3018             :                                             std::numeric_limits<double>::max()),
    3019             :               0);
    3020           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3021             :                   GDT_Int16, std::numeric_limits<int16_t>::lowest()),
    3022             :               std::numeric_limits<int16_t>::lowest() + 1);
    3023           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3024             :                   GDT_Int16, std::numeric_limits<int16_t>::max()),
    3025             :               std::numeric_limits<int16_t>::max() - 1);
    3026             : 
    3027             :     // Test GDT_UInt32
    3028           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3029             :                   GDT_UInt32, std::numeric_limits<double>::lowest()),
    3030             :               0);
    3031           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_UInt32,
    3032             :                                             std::numeric_limits<double>::max()),
    3033             :               0);
    3034           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3035             :                   GDT_UInt32, std::numeric_limits<uint32_t>::lowest()),
    3036             :               std::numeric_limits<uint32_t>::lowest() + 1);
    3037           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3038             :                   GDT_UInt32, std::numeric_limits<uint32_t>::max()),
    3039             :               std::numeric_limits<uint32_t>::max() - 1);
    3040             : 
    3041             :     // Test GDT_Int32
    3042           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3043             :                   GDT_Int32, std::numeric_limits<double>::lowest()),
    3044             :               0);
    3045           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Int32,
    3046             :                                             std::numeric_limits<double>::max()),
    3047             :               0);
    3048           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3049             :                   GDT_Int32, std::numeric_limits<int32_t>::lowest()),
    3050             :               std::numeric_limits<int32_t>::lowest() + 1);
    3051           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3052             :                   GDT_Int32, std::numeric_limits<int32_t>::max()),
    3053             :               std::numeric_limits<int32_t>::max() - 1);
    3054             : 
    3055             :     // Test GDT_UInt64
    3056           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3057             :                   GDT_UInt64, std::numeric_limits<double>::lowest()),
    3058             :               0);
    3059           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_UInt64,
    3060             :                                             std::numeric_limits<double>::max()),
    3061             :               0);
    3062           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3063             :                   GDT_UInt64,
    3064             :                   static_cast<double>(std::numeric_limits<uint64_t>::lowest())),
    3065             :               static_cast<double>(std::numeric_limits<uint64_t>::lowest()) + 1);
    3066             :     // uin64_t max is not representable in double so we expect the next value to be returned
    3067           1 :     EXPECT_EQ(
    3068             :         GDALGetNoDataReplacementValue(
    3069             :             GDT_UInt64,
    3070             :             static_cast<double>(std::numeric_limits<uint64_t>::max())),
    3071             :         std::nextafter(
    3072             :             static_cast<double>(std::numeric_limits<uint64_t>::max()), 0) -
    3073             :             1);
    3074             : 
    3075             :     // Test GDT_Int64
    3076           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3077             :                   GDT_Int64, std::numeric_limits<double>::lowest()),
    3078             :               0);
    3079           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Int64,
    3080             :                                             std::numeric_limits<double>::max()),
    3081             :               0);
    3082             :     // in64_t max is not representable in double so we expect the next value to be returned
    3083           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3084             :                   GDT_Int64,
    3085             :                   static_cast<double>(std::numeric_limits<int64_t>::lowest())),
    3086             :               static_cast<double>(std::numeric_limits<int64_t>::lowest()) + 1);
    3087           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3088             :                   GDT_Int64,
    3089             :                   static_cast<double>(std::numeric_limits<int64_t>::max())),
    3090             :               std::nextafter(
    3091             :                   static_cast<double>(std::numeric_limits<int64_t>::max()), 0) -
    3092             :                   1);
    3093             : 
    3094             :     // Test floating point types
    3095             : 
    3096             :     // out of range for float32
    3097           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3098             :                   GDT_Float32, std::numeric_limits<double>::lowest()),
    3099             :               0.0);
    3100           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Float32,
    3101             :                                             std::numeric_limits<double>::max()),
    3102             :               0.0);
    3103           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3104             :                   GDT_Float32, std::numeric_limits<double>::infinity()),
    3105             :               0.0);
    3106           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3107             :                   GDT_Float32, -std::numeric_limits<double>::infinity()),
    3108             :               0.0);
    3109             : 
    3110             :     // in range for float 32
    3111           1 :     EXPECT_EQ(
    3112             :         static_cast<float>(GDALGetNoDataReplacementValue(GDT_Float32, -1.0)),
    3113             :         std::nextafter(float(-1.0), 0.0f));
    3114           1 :     EXPECT_EQ(
    3115             :         static_cast<float>(GDALGetNoDataReplacementValue(GDT_Float32, 1.1)),
    3116             :         std::nextafter(float(1.1), 2.0f));
    3117           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3118             :                   GDT_Float32, std::numeric_limits<float>::lowest()),
    3119             :               std::nextafter(std::numeric_limits<float>::lowest(), 0.0f));
    3120             : 
    3121           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Float32,
    3122             :                                             std::numeric_limits<float>::max()),
    3123             :               static_cast<double>(
    3124             :                   std::nextafter(std::numeric_limits<float>::max(), 0.0f)));
    3125             : 
    3126             :     // in range for float64
    3127           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3128             :                   GDT_Float64, std::numeric_limits<double>::lowest()),
    3129             :               std::nextafter(std::numeric_limits<double>::lowest(), 0.0));
    3130           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Float64,
    3131             :                                             std::numeric_limits<double>::max()),
    3132             :               std::nextafter(std::numeric_limits<double>::max(), 0.0));
    3133             : 
    3134           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3135             :                   GDT_Float64, std::numeric_limits<double>::lowest()),
    3136             :               std::nextafter(std::numeric_limits<double>::lowest(), 0.0));
    3137           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Float64,
    3138             :                                             std::numeric_limits<double>::max()),
    3139             :               std::nextafter(std::numeric_limits<double>::max(), 0.0));
    3140             : 
    3141           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Float64, double(-1.0)),
    3142             :               std::nextafter(double(-1.0), 0.0));
    3143           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(GDT_Float64, double(1.1)),
    3144             :               std::nextafter(double(1.1), 2.0));
    3145             : 
    3146             :     // test infinity
    3147           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3148             :                   GDT_Float64, std::numeric_limits<double>::infinity()),
    3149             :               0.0);
    3150           1 :     EXPECT_EQ(GDALGetNoDataReplacementValue(
    3151             :                   GDT_Float64, -std::numeric_limits<double>::infinity()),
    3152             :               0.0);
    3153           1 : }
    3154             : 
    3155             : // Test GDALRasterBand::GetIndexColorTranslationTo()
    3156           4 : TEST_F(test_gdal, GetIndexColorTranslationTo)
    3157             : {
    3158             :     GDALDatasetUniquePtr poSrcDS(
    3159             :         GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    3160           2 :             ->Create("", 1, 1, 1, GDT_Byte, nullptr));
    3161             :     {
    3162           2 :         GDALColorTable oCT;
    3163             :         {
    3164             :             GDALColorEntry e;
    3165           1 :             e.c1 = 0;
    3166           1 :             e.c2 = 0;
    3167           1 :             e.c3 = 0;
    3168           1 :             e.c4 = 255;
    3169           1 :             oCT.SetColorEntry(0, &e);
    3170             :         }
    3171             :         {
    3172             :             GDALColorEntry e;
    3173           1 :             e.c1 = 1;
    3174           1 :             e.c2 = 0;
    3175           1 :             e.c3 = 0;
    3176           1 :             e.c4 = 255;
    3177           1 :             oCT.SetColorEntry(1, &e);
    3178             :         }
    3179             :         {
    3180             :             GDALColorEntry e;
    3181           1 :             e.c1 = 255;
    3182           1 :             e.c2 = 255;
    3183           1 :             e.c3 = 255;
    3184           1 :             e.c4 = 255;
    3185           1 :             oCT.SetColorEntry(2, &e);
    3186             :         }
    3187             :         {
    3188             :             GDALColorEntry e;
    3189           1 :             e.c1 = 125;
    3190           1 :             e.c2 = 126;
    3191           1 :             e.c3 = 127;
    3192           1 :             e.c4 = 0;
    3193           1 :             oCT.SetColorEntry(3, &e);
    3194           1 :             poSrcDS->GetRasterBand(1)->SetNoDataValue(3);
    3195             :         }
    3196           1 :         poSrcDS->GetRasterBand(1)->SetColorTable(&oCT);
    3197             :     }
    3198             : 
    3199             :     GDALDatasetUniquePtr poDstDS(
    3200             :         GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    3201           2 :             ->Create("", 1, 1, 1, GDT_Byte, nullptr));
    3202             :     {
    3203           2 :         GDALColorTable oCT;
    3204             :         {
    3205             :             GDALColorEntry e;
    3206           1 :             e.c1 = 255;
    3207           1 :             e.c2 = 255;
    3208           1 :             e.c3 = 255;
    3209           1 :             e.c4 = 255;
    3210           1 :             oCT.SetColorEntry(0, &e);
    3211             :         }
    3212             :         {
    3213             :             GDALColorEntry e;
    3214           1 :             e.c1 = 0;
    3215           1 :             e.c2 = 0;
    3216           1 :             e.c3 = 1;
    3217           1 :             e.c4 = 255;
    3218           1 :             oCT.SetColorEntry(1, &e);
    3219             :         }
    3220             :         {
    3221             :             GDALColorEntry e;
    3222           1 :             e.c1 = 12;
    3223           1 :             e.c2 = 13;
    3224           1 :             e.c3 = 14;
    3225           1 :             e.c4 = 0;
    3226           1 :             oCT.SetColorEntry(2, &e);
    3227           1 :             poSrcDS->GetRasterBand(1)->SetNoDataValue(2);
    3228             :         }
    3229           1 :         poDstDS->GetRasterBand(1)->SetColorTable(&oCT);
    3230             :     }
    3231             : 
    3232             :     unsigned char *panTranslationTable =
    3233           1 :         poSrcDS->GetRasterBand(1)->GetIndexColorTranslationTo(
    3234             :             poDstDS->GetRasterBand(1));
    3235           1 :     EXPECT_EQ(static_cast<int>(panTranslationTable[0]), 1);
    3236           1 :     EXPECT_EQ(static_cast<int>(panTranslationTable[1]), 1);
    3237           1 :     EXPECT_EQ(static_cast<int>(panTranslationTable[2]), 0);
    3238           1 :     EXPECT_EQ(static_cast<int>(panTranslationTable[3]),
    3239             :               2);  // special nodata mapping
    3240           1 :     CPLFree(panTranslationTable);
    3241           1 : }
    3242             : 
    3243             : // Test effect of MarkSuppressOnClose() with the final FlushCache() at dataset
    3244             : // destruction
    3245           4 : TEST_F(test_gdal, MarkSuppressOnClose)
    3246             : {
    3247           1 :     const char *pszFilename = "/vsimem/out.tif";
    3248           1 :     const char *const apszOptions[] = {"PROFILE=BASELINE", nullptr};
    3249           1 :     auto hDrv = GDALGetDriverByName("GTiff");
    3250           1 :     if (!hDrv)
    3251             :     {
    3252           0 :         GTEST_SKIP() << "GTiff driver missing";
    3253             :     }
    3254             :     else
    3255             :     {
    3256             :         GDALDatasetUniquePtr poDstDS(GDALDriver::FromHandle(hDrv)->Create(
    3257           2 :             pszFilename, 1, 1, 1, GDT_Byte, apszOptions));
    3258           1 :         poDstDS->SetMetadataItem("FOO", "BAR");
    3259           1 :         poDstDS->MarkSuppressOnClose();
    3260           1 :         poDstDS->GetRasterBand(1)->Fill(255);
    3261           1 :         poDstDS->FlushCache(true);
    3262             :         // All buffers have been flushed, but our dirty block should not have
    3263             :         // been written hence the checksum will be 0
    3264           1 :         EXPECT_EQ(GDALChecksumImage(
    3265             :                       GDALRasterBand::FromHandle(poDstDS->GetRasterBand(1)), 0,
    3266             :                       0, 1, 1),
    3267             :                   0);
    3268             :     }
    3269             :     {
    3270             :         VSIStatBufL sStat;
    3271           1 :         EXPECT_TRUE(VSIStatL(CPLSPrintf("%s.aux.xml", pszFilename), &sStat) !=
    3272             :                     0);
    3273             :     }
    3274             : }
    3275             : 
    3276             : // Test effect of UnMarkSuppressOnClose()
    3277           4 : TEST_F(test_gdal, UnMarkSuppressOnClose)
    3278             : {
    3279           1 :     const char *pszFilename = "/vsimem/out.tif";
    3280           1 :     const char *const apszOptions[] = {"PROFILE=BASELINE", nullptr};
    3281           1 :     auto hDrv = GDALGetDriverByName("GTiff");
    3282           1 :     if (!hDrv)
    3283             :     {
    3284           0 :         GTEST_SKIP() << "GTiff driver missing";
    3285             :     }
    3286             :     else
    3287             :     {
    3288             :         GDALDatasetUniquePtr poDstDS(GDALDriver::FromHandle(hDrv)->Create(
    3289           2 :             pszFilename, 1, 1, 1, GDT_Byte, apszOptions));
    3290           1 :         poDstDS->MarkSuppressOnClose();
    3291           1 :         poDstDS->GetRasterBand(1)->Fill(255);
    3292           1 :         if (poDstDS->IsMarkedSuppressOnClose())
    3293           1 :             poDstDS->UnMarkSuppressOnClose();
    3294           1 :         poDstDS->FlushCache(true);
    3295             :         // All buffers have been flushed, and our dirty block should have
    3296             :         // been written hence the checksum will not be 0
    3297           1 :         EXPECT_NE(GDALChecksumImage(
    3298             :                       GDALRasterBand::FromHandle(poDstDS->GetRasterBand(1)), 0,
    3299             :                       0, 1, 1),
    3300             :                   0);
    3301             :         VSIStatBufL sStat;
    3302           1 :         EXPECT_TRUE(VSIStatL(pszFilename, &sStat) == 0);
    3303           1 :         VSIUnlink(pszFilename);
    3304             :     }
    3305             : }
    3306             : 
    3307          11 : template <class T> void TestCachedPixelAccessor()
    3308             : {
    3309          11 :     constexpr auto eType = GDALCachedPixelAccessorGetDataType<T>::DataType;
    3310          22 :     auto poDS = std::unique_ptr<GDALDataset>(
    3311             :         GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    3312             :             ->Create("", 11, 23, 1, eType, nullptr));
    3313          11 :     auto poBand = poDS->GetRasterBand(1);
    3314          22 :     GDALCachedPixelAccessor<T, 4> accessor(poBand);
    3315         264 :     for (int iY = 0; iY < poBand->GetYSize(); iY++)
    3316             :     {
    3317        3036 :         for (int iX = 0; iX < poBand->GetXSize(); iX++)
    3318             :         {
    3319        2783 :             accessor.Set(iX, iY, static_cast<T>(iY * poBand->GetXSize() + iX));
    3320             :         }
    3321             :     }
    3322         264 :     for (int iY = 0; iY < poBand->GetYSize(); iY++)
    3323             :     {
    3324        3036 :         for (int iX = 0; iX < poBand->GetXSize(); iX++)
    3325             :         {
    3326        2783 :             EXPECT_EQ(accessor.Get(iX, iY),
    3327             :                       static_cast<T>(iY * poBand->GetXSize() + iX));
    3328             :         }
    3329             :     }
    3330             : 
    3331          22 :     std::vector<T> values(static_cast<size_t>(poBand->GetYSize()) *
    3332          11 :                           poBand->GetXSize());
    3333          11 :     accessor.FlushCache();
    3334          11 :     EXPECT_EQ(poBand->RasterIO(GF_Read, 0, 0, poBand->GetXSize(),
    3335             :                                poBand->GetYSize(), values.data(),
    3336             :                                poBand->GetXSize(), poBand->GetYSize(), eType, 0,
    3337             :                                0, nullptr),
    3338             :               CE_None);
    3339         264 :     for (int iY = 0; iY < poBand->GetYSize(); iY++)
    3340             :     {
    3341        3036 :         for (int iX = 0; iX < poBand->GetXSize(); iX++)
    3342             :         {
    3343        2783 :             EXPECT_EQ(values[iY * poBand->GetXSize() + iX],
    3344             :                       static_cast<T>(iY * poBand->GetXSize() + iX));
    3345             :         }
    3346             :     }
    3347          11 : }
    3348             : 
    3349             : // Test GDALCachedPixelAccessor
    3350           4 : TEST_F(test_gdal, GDALCachedPixelAccessor)
    3351             : {
    3352           1 :     TestCachedPixelAccessor<GByte>();
    3353           1 :     TestCachedPixelAccessor<GUInt16>();
    3354           1 :     TestCachedPixelAccessor<GInt16>();
    3355           1 :     TestCachedPixelAccessor<GUInt32>();
    3356           1 :     TestCachedPixelAccessor<GInt32>();
    3357           1 :     TestCachedPixelAccessor<GUInt64>();
    3358           1 :     TestCachedPixelAccessor<GInt64>();
    3359           1 :     TestCachedPixelAccessor<uint64_t>();
    3360           1 :     TestCachedPixelAccessor<int64_t>();
    3361           1 :     TestCachedPixelAccessor<float>();
    3362           1 :     TestCachedPixelAccessor<double>();
    3363           1 : }
    3364             : 
    3365             : // Test VRT and caching of sources w.r.t open options
    3366             : // (https://github.com/OSGeo/gdal/issues/5989)
    3367           4 : TEST_F(test_gdal, VRTCachingOpenOptions)
    3368             : {
    3369           1 :     if (GDALGetMetadataItem(GDALGetDriverByName("VRT"), GDAL_DMD_OPENOPTIONLIST,
    3370           1 :                             nullptr) == nullptr)
    3371             :     {
    3372           0 :         GTEST_SKIP() << "VRT driver Open() missing";
    3373             :     }
    3374             : 
    3375             :     class TestRasterBand : public GDALRasterBand
    3376             :     {
    3377             :       protected:
    3378           3 :         CPLErr IReadBlock(int, int, void *pImage) override
    3379             :         {
    3380           3 :             static_cast<GByte *>(pImage)[0] = 0;
    3381           3 :             return CE_None;
    3382             :         }
    3383             : 
    3384             :       public:
    3385           3 :         TestRasterBand()
    3386           3 :         {
    3387           3 :             nBlockXSize = 1;
    3388           3 :             nBlockYSize = 1;
    3389           3 :             eDataType = GDT_Byte;
    3390           3 :         }
    3391             :     };
    3392             : 
    3393             :     static int nCountZeroOpenOptions = 0;
    3394             :     static int nCountWithOneOpenOptions = 0;
    3395             : 
    3396             :     class TestDataset : public GDALDataset
    3397             :     {
    3398             :       public:
    3399           3 :         TestDataset()
    3400           3 :         {
    3401           3 :             nRasterXSize = 1;
    3402           3 :             nRasterYSize = 1;
    3403           3 :             SetBand(1, new TestRasterBand());
    3404           3 :         }
    3405             : 
    3406           3 :         static GDALDataset *TestOpen(GDALOpenInfo *poOpenInfo)
    3407             :         {
    3408           3 :             if (strcmp(poOpenInfo->pszFilename, ":::DUMMY:::") != 0)
    3409           0 :                 return nullptr;
    3410           3 :             if (poOpenInfo->papszOpenOptions == nullptr)
    3411           1 :                 nCountZeroOpenOptions++;
    3412             :             else
    3413           2 :                 nCountWithOneOpenOptions++;
    3414           3 :             return new TestDataset();
    3415             :         }
    3416             :     };
    3417             : 
    3418           2 :     std::unique_ptr<GDALDriver> driver(new GDALDriver());
    3419           1 :     driver->SetDescription("TEST_VRT_SOURCE_OPEN_OPTION");
    3420           1 :     driver->pfnOpen = TestDataset::TestOpen;
    3421           1 :     GetGDALDriverManager()->RegisterDriver(driver.get());
    3422             : 
    3423           1 :     const char *pszVRT = R"(
    3424             : <VRTDataset rasterXSize="1" rasterYSize="1">
    3425             :   <VRTRasterBand dataType="Byte" band="1" subClass="VRTSourcedRasterBand">
    3426             :     <SimpleSource>
    3427             :       <SourceFilename relativeToVRT="0">:::DUMMY:::</SourceFilename>
    3428             :     </SimpleSource>
    3429             :     <SimpleSource>
    3430             :       <SourceFilename relativeToVRT="0">:::DUMMY:::</SourceFilename>
    3431             :     </SimpleSource>
    3432             :     <SimpleSource>
    3433             :       <SourceFilename relativeToVRT="0">:::DUMMY:::</SourceFilename>
    3434             :       <OpenOptions>
    3435             :           <OOI key="TESTARG">present</OOI>
    3436             :       </OpenOptions>
    3437             :     </SimpleSource>
    3438             :     <SimpleSource>
    3439             :       <SourceFilename relativeToVRT="0">:::DUMMY:::</SourceFilename>
    3440             :       <OpenOptions>
    3441             :           <OOI key="TESTARG">present</OOI>
    3442             :       </OpenOptions>
    3443             :     </SimpleSource>
    3444             :     <SimpleSource>
    3445             :       <SourceFilename relativeToVRT="0">:::DUMMY:::</SourceFilename>
    3446             :       <OpenOptions>
    3447             :           <OOI key="TESTARG">another_one</OOI>
    3448             :       </OpenOptions>
    3449             :     </SimpleSource>
    3450             :   </VRTRasterBand>
    3451             : </VRTDataset>)";
    3452           2 :     auto ds = std::unique_ptr<GDALDataset>(GDALDataset::Open(pszVRT));
    3453             : 
    3454             :     // Trigger reading data, which triggers opening of source datasets
    3455           1 :     auto rb = ds->GetRasterBand(1);
    3456             :     double minmax[2];
    3457           1 :     GDALComputeRasterMinMax(GDALRasterBand::ToHandle(rb), TRUE, minmax);
    3458             : 
    3459           1 :     ds.reset();
    3460           1 :     GetGDALDriverManager()->DeregisterDriver(driver.get());
    3461             : 
    3462           1 :     EXPECT_EQ(nCountZeroOpenOptions, 1);
    3463           1 :     EXPECT_EQ(nCountWithOneOpenOptions, 2);
    3464             : }
    3465             : 
    3466             : // Test GDALDeinterleave 3 components Byte()
    3467           4 : TEST_F(test_gdal, GDALDeinterleave3ComponentsByte)
    3468             : {
    3469           1 :     GByte *pabySrc = static_cast<GByte *>(CPLMalloc(3 * 4 * 15));
    3470         181 :     for (int i = 0; i < 3 * 4 * 15; i++)
    3471         180 :         pabySrc[i] = static_cast<GByte>(i);
    3472           1 :     GByte *pabyDest0 = static_cast<GByte *>(CPLMalloc(4 * 15));
    3473           1 :     GByte *pabyDest1 = static_cast<GByte *>(CPLMalloc(4 * 15));
    3474           1 :     GByte *pabyDest2 = static_cast<GByte *>(CPLMalloc(4 * 15));
    3475           1 :     void *ppabyDest[] = {pabyDest0, pabyDest1, pabyDest2};
    3476           3 :     for (int nIters : {1, 4 * 15})
    3477             :     {
    3478           2 :         GDALDeinterleave(pabySrc, GDT_Byte, 3, ppabyDest, GDT_Byte, nIters);
    3479          63 :         for (int i = 0; i < nIters; i++)
    3480             :         {
    3481          61 :             EXPECT_EQ(pabyDest0[i], 3 * i);
    3482          61 :             EXPECT_EQ(pabyDest1[i], 3 * i + 1);
    3483          61 :             EXPECT_EQ(pabyDest2[i], 3 * i + 2);
    3484             :         }
    3485             :     }
    3486           1 :     VSIFree(pabySrc);
    3487           1 :     VSIFree(pabyDest0);
    3488           1 :     VSIFree(pabyDest1);
    3489           1 :     VSIFree(pabyDest2);
    3490           1 : }
    3491             : 
    3492             : // Test GDALDeinterleave 3 components Byte() without SSSE3
    3493           4 : TEST_F(test_gdal, GDALDeinterleave3ComponentsByte_NOSSE3)
    3494             : {
    3495           1 :     GByte *pabySrc = static_cast<GByte *>(CPLMalloc(3 * 4 * 15));
    3496         181 :     for (int i = 0; i < 3 * 4 * 15; i++)
    3497         180 :         pabySrc[i] = static_cast<GByte>(i);
    3498           1 :     GByte *pabyDest0 = static_cast<GByte *>(CPLMalloc(4 * 15));
    3499           1 :     GByte *pabyDest1 = static_cast<GByte *>(CPLMalloc(4 * 15));
    3500           1 :     GByte *pabyDest2 = static_cast<GByte *>(CPLMalloc(4 * 15));
    3501           1 :     void *ppabyDest[] = {pabyDest0, pabyDest1, pabyDest2};
    3502           3 :     for (int nIters : {1, 4 * 15})
    3503             :     {
    3504           2 :         CPLSetConfigOption("GDAL_USE_SSSE3", "NO");
    3505           2 :         GDALDeinterleave(pabySrc, GDT_Byte, 3, ppabyDest, GDT_Byte, nIters);
    3506           2 :         CPLSetConfigOption("GDAL_USE_SSSE3", nullptr);
    3507          63 :         for (int i = 0; i < nIters; i++)
    3508             :         {
    3509          61 :             EXPECT_EQ(pabyDest0[i], 3 * i);
    3510          61 :             EXPECT_EQ(pabyDest1[i], 3 * i + 1);
    3511          61 :             EXPECT_EQ(pabyDest2[i], 3 * i + 2);
    3512             :         }
    3513             :     }
    3514           1 :     VSIFree(pabySrc);
    3515           1 :     VSIFree(pabyDest0);
    3516           1 :     VSIFree(pabyDest1);
    3517           1 :     VSIFree(pabyDest2);
    3518           1 : }
    3519             : 
    3520             : // Test GDALDeinterleave 4 components Byte()
    3521           4 : TEST_F(test_gdal, GDALDeinterleave4ComponentsByte)
    3522             : {
    3523           1 :     GByte *pabySrc = static_cast<GByte *>(CPLMalloc(3 * 4 * 15));
    3524         181 :     for (int i = 0; i < 3 * 4 * 15; i++)
    3525         180 :         pabySrc[i] = static_cast<GByte>(i);
    3526           1 :     GByte *pabyDest0 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3527           1 :     GByte *pabyDest1 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3528           1 :     GByte *pabyDest2 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3529           1 :     GByte *pabyDest3 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3530           1 :     void *ppabyDest[] = {pabyDest0, pabyDest1, pabyDest2, pabyDest3};
    3531           3 :     for (int nIters : {1, 3 * 15})
    3532             :     {
    3533           2 :         GDALDeinterleave(pabySrc, GDT_Byte, 4, ppabyDest, GDT_Byte, nIters);
    3534          48 :         for (int i = 0; i < nIters; i++)
    3535             :         {
    3536          46 :             EXPECT_EQ(pabyDest0[i], 4 * i);
    3537          46 :             EXPECT_EQ(pabyDest1[i], 4 * i + 1);
    3538          46 :             EXPECT_EQ(pabyDest2[i], 4 * i + 2);
    3539          46 :             EXPECT_EQ(pabyDest3[i], 4 * i + 3);
    3540             :         }
    3541             :     }
    3542           1 :     VSIFree(pabySrc);
    3543           1 :     VSIFree(pabyDest0);
    3544           1 :     VSIFree(pabyDest1);
    3545           1 :     VSIFree(pabyDest2);
    3546           1 :     VSIFree(pabyDest3);
    3547           1 : }
    3548             : 
    3549             : // Test GDALDeinterleave 4 components Byte without SSSE3
    3550           4 : TEST_F(test_gdal, GDALDeinterleave4ComponentsByte_NOSSE3)
    3551             : {
    3552           1 :     GByte *pabySrc = static_cast<GByte *>(CPLMalloc(3 * 4 * 15));
    3553         181 :     for (int i = 0; i < 3 * 4 * 15; i++)
    3554         180 :         pabySrc[i] = static_cast<GByte>(i);
    3555           1 :     GByte *pabyDest0 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3556           1 :     GByte *pabyDest1 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3557           1 :     GByte *pabyDest2 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3558           1 :     GByte *pabyDest3 = static_cast<GByte *>(CPLMalloc(3 * 15));
    3559           1 :     void *ppabyDest[] = {pabyDest0, pabyDest1, pabyDest2, pabyDest3};
    3560           3 :     for (int nIters : {1, 3 * 15})
    3561             :     {
    3562           2 :         CPLSetConfigOption("GDAL_USE_SSSE3", "NO");
    3563           2 :         GDALDeinterleave(pabySrc, GDT_Byte, 4, ppabyDest, GDT_Byte, nIters);
    3564           2 :         CPLSetConfigOption("GDAL_USE_SSSE3", nullptr);
    3565          48 :         for (int i = 0; i < nIters; i++)
    3566             :         {
    3567          46 :             EXPECT_EQ(pabyDest0[i], 4 * i);
    3568          46 :             EXPECT_EQ(pabyDest1[i], 4 * i + 1);
    3569          46 :             EXPECT_EQ(pabyDest2[i], 4 * i + 2);
    3570          46 :             EXPECT_EQ(pabyDest3[i], 4 * i + 3);
    3571             :         }
    3572             :     }
    3573           1 :     VSIFree(pabySrc);
    3574           1 :     VSIFree(pabyDest0);
    3575           1 :     VSIFree(pabyDest1);
    3576           1 :     VSIFree(pabyDest2);
    3577           1 :     VSIFree(pabyDest3);
    3578           1 : }
    3579             : 
    3580             : // Test GDALDeinterleave general case
    3581           4 : TEST_F(test_gdal, GDALDeinterleaveGeneralCase)
    3582             : {
    3583           1 :     GByte *pabySrc = static_cast<GByte *>(CPLMalloc(3 * 2));
    3584           7 :     for (int i = 0; i < 3 * 2; i++)
    3585           6 :         pabySrc[i] = static_cast<GByte>(i);
    3586           1 :     GUInt16 *panDest0 = static_cast<GUInt16 *>(CPLMalloc(3 * sizeof(uint16_t)));
    3587           1 :     GUInt16 *panDest1 = static_cast<GUInt16 *>(CPLMalloc(3 * sizeof(uint16_t)));
    3588           1 :     void *ppanDest[] = {panDest0, panDest1};
    3589           1 :     GDALDeinterleave(pabySrc, GDT_Byte, 2, ppanDest, GDT_UInt16, 3);
    3590           4 :     for (int i = 0; i < 3; i++)
    3591             :     {
    3592           3 :         EXPECT_EQ(panDest0[i], 2 * i);
    3593           3 :         EXPECT_EQ(panDest1[i], 2 * i + 1);
    3594             :     }
    3595           1 :     VSIFree(pabySrc);
    3596           1 :     VSIFree(panDest0);
    3597           1 :     VSIFree(panDest1);
    3598           1 : }
    3599             : 
    3600             : // Test GDALDeinterleave 3 components UInt16()
    3601           4 : TEST_F(test_gdal, GDALDeinterleave3ComponentsUInt16)
    3602             : {
    3603             :     GUInt16 *panSrc =
    3604           1 :         static_cast<GUInt16 *>(CPLMalloc(3 * 4 * 15 * sizeof(GUInt16)));
    3605         181 :     for (int i = 0; i < 3 * 4 * 15; i++)
    3606         180 :         panSrc[i] = static_cast<GUInt16>(i + 32767);
    3607             :     GUInt16 *panDest0 =
    3608           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3609             :     GUInt16 *panDest1 =
    3610           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3611             :     GUInt16 *panDest2 =
    3612           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3613           1 :     void *ppanDest[] = {panDest0, panDest1, panDest2};
    3614           3 :     for (int nIters : {1, 4 * 15})
    3615             :     {
    3616           2 :         GDALDeinterleave(panSrc, GDT_UInt16, 3, ppanDest, GDT_UInt16, nIters);
    3617          63 :         for (int i = 0; i < nIters; i++)
    3618             :         {
    3619          61 :             EXPECT_EQ(panDest0[i], 3 * i + 32767);
    3620          61 :             EXPECT_EQ(panDest1[i], 3 * i + 1 + 32767);
    3621          61 :             EXPECT_EQ(panDest2[i], 3 * i + 2 + 32767);
    3622             :         }
    3623             :     }
    3624           1 :     VSIFree(panSrc);
    3625           1 :     VSIFree(panDest0);
    3626           1 :     VSIFree(panDest1);
    3627           1 :     VSIFree(panDest2);
    3628           1 : }
    3629             : 
    3630             : // Test GDALDeinterleave 4 components UInt16()
    3631           4 : TEST_F(test_gdal, GDALDeinterleave4ComponentsUInt16)
    3632             : {
    3633             :     GUInt16 *panSrc =
    3634           1 :         static_cast<GUInt16 *>(CPLMalloc(3 * 4 * 15 * sizeof(GUInt16)));
    3635         181 :     for (int i = 0; i < 3 * 4 * 15; i++)
    3636         180 :         panSrc[i] = static_cast<GUInt16>(i + 32767);
    3637             :     GUInt16 *panDest0 =
    3638           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3639             :     GUInt16 *panDest1 =
    3640           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3641             :     GUInt16 *panDest2 =
    3642           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3643             :     GUInt16 *panDest3 =
    3644           1 :         static_cast<GUInt16 *>(CPLMalloc(4 * 15 * sizeof(GUInt16)));
    3645           1 :     void *ppanDest[] = {panDest0, panDest1, panDest2, panDest3};
    3646           3 :     for (int nIters : {1, 3 * 15})
    3647             :     {
    3648           2 :         GDALDeinterleave(panSrc, GDT_UInt16, 4, ppanDest, GDT_UInt16, nIters);
    3649          48 :         for (int i = 0; i < nIters; i++)
    3650             :         {
    3651          46 :             EXPECT_EQ(panDest0[i], 4 * i + 32767);
    3652          46 :             EXPECT_EQ(panDest1[i], 4 * i + 1 + 32767);
    3653          46 :             EXPECT_EQ(panDest2[i], 4 * i + 2 + 32767);
    3654          46 :             EXPECT_EQ(panDest3[i], 4 * i + 3 + 32767);
    3655             :         }
    3656             :     }
    3657           1 :     VSIFree(panSrc);
    3658           1 :     VSIFree(panDest0);
    3659           1 :     VSIFree(panDest1);
    3660           1 :     VSIFree(panDest2);
    3661           1 :     VSIFree(panDest3);
    3662           1 : }
    3663             : 
    3664             : // Test GDALDataset::ReportError()
    3665           4 : TEST_F(test_gdal, GDALDatasetReportError)
    3666             : {
    3667             :     GDALDatasetUniquePtr poSrcDS(
    3668             :         GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    3669           2 :             ->Create("", 1, 1, 1, GDT_Byte, nullptr));
    3670             : 
    3671           1 :     CPLPushErrorHandler(CPLQuietErrorHandler);
    3672           1 :     poSrcDS->ReportError("foo", CE_Warning, CPLE_AppDefined, "bar");
    3673           1 :     CPLPopErrorHandler();
    3674           1 :     EXPECT_STREQ(CPLGetLastErrorMsg(), "foo: bar");
    3675             : 
    3676           1 :     CPLPushErrorHandler(CPLQuietErrorHandler);
    3677           1 :     poSrcDS->ReportError("%foo", CE_Warning, CPLE_AppDefined, "bar");
    3678           1 :     CPLPopErrorHandler();
    3679           1 :     EXPECT_STREQ(CPLGetLastErrorMsg(), "%foo: bar");
    3680             : 
    3681           1 :     CPLPushErrorHandler(CPLQuietErrorHandler);
    3682           1 :     poSrcDS->ReportError(
    3683             :         "this_is_"
    3684             :         "wayyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3685             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3686             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3687             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3688             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3689             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3690             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3691             :         "yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy"
    3692             :         "yyyyyyy_too_long/foo",
    3693           1 :         CE_Warning, CPLE_AppDefined, "bar");
    3694           1 :     CPLPopErrorHandler();
    3695           1 :     EXPECT_STREQ(CPLGetLastErrorMsg(), "foo: bar");
    3696           1 : }
    3697             : 
    3698             : // Test GDALDataset::GetCompressionFormats() and ReadCompressedData()
    3699           4 : TEST_F(test_gdal, gtiff_ReadCompressedData)
    3700             : {
    3701           1 :     if (!GDALGetDriverByName("GTiff"))
    3702             :     {
    3703           0 :         GTEST_SKIP() << "GTiff driver missing";
    3704             :     }
    3705           1 :     if (GDALGetDriverByName("JPEG") == nullptr)
    3706             :     {
    3707           0 :         GTEST_SKIP() << "JPEG support missing";
    3708             :     }
    3709             : 
    3710             :     GDALDatasetUniquePtr poSrcDS(GDALDataset::FromHandle(
    3711           1 :         GDALDataset::Open((tut::common::data_basedir +
    3712             :                            "/../../gcore/data/byte_jpg_unusual_jpegtable.tif")
    3713           1 :                               .c_str())));
    3714           1 :     ASSERT_TRUE(poSrcDS);
    3715             : 
    3716             :     const CPLStringList aosRet(GDALDatasetGetCompressionFormats(
    3717           1 :         GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 20, 20, 1, nullptr));
    3718           1 :     EXPECT_EQ(aosRet.size(), 1);
    3719           1 :     if (aosRet.size() == 1)
    3720             :     {
    3721           1 :         EXPECT_STREQ(aosRet[0], "JPEG");
    3722             :     }
    3723             : 
    3724             :     {
    3725           1 :         int nBand = 1;
    3726           1 :         EXPECT_EQ(CPLStringList(GDALDatasetGetCompressionFormats(
    3727             :                                     GDALDataset::ToHandle(poSrcDS.get()), 0, 0,
    3728             :                                     20, 20, 1, &nBand))
    3729             :                       .size(),
    3730             :                   1);
    3731             :     }
    3732             : 
    3733             :     // nBandCout > nBands
    3734           1 :     EXPECT_EQ(CPLStringList(GDALDatasetGetCompressionFormats(
    3735             :                                 GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 20,
    3736             :                                 20, 2, nullptr))
    3737             :                   .size(),
    3738             :               0);
    3739             : 
    3740             :     // Cannot subset just one pixel
    3741           1 :     EXPECT_EQ(CPLStringList(GDALDatasetGetCompressionFormats(
    3742             :                                 GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 1,
    3743             :                                 1, 1, nullptr))
    3744             :                   .size(),
    3745             :               0);
    3746             : 
    3747             :     // Wrong band number
    3748             :     {
    3749           1 :         int nBand = 2;
    3750           1 :         EXPECT_EQ(CPLStringList(GDALDatasetGetCompressionFormats(
    3751             :                                     GDALDataset::ToHandle(poSrcDS.get()), 0, 0,
    3752             :                                     20, 20, 1, &nBand))
    3753             :                       .size(),
    3754             :                   0);
    3755             :     }
    3756             : 
    3757           1 :     EXPECT_EQ(GDALDatasetReadCompressedData(
    3758             :                   GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20, 20, 1,
    3759             :                   nullptr, nullptr, nullptr, nullptr),
    3760             :               CE_None);
    3761             : 
    3762             :     size_t nNeededSize;
    3763             :     {
    3764           1 :         char *pszDetailedFormat = nullptr;
    3765           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3766             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20,
    3767             :                       20, 1, nullptr, nullptr, &nNeededSize,
    3768             :                       &pszDetailedFormat),
    3769             :                   CE_None);
    3770           1 :         EXPECT_EQ(nNeededSize, 476);
    3771           1 :         EXPECT_TRUE(pszDetailedFormat != nullptr);
    3772           1 :         if (pszDetailedFormat)
    3773             :         {
    3774           1 :             ASSERT_STREQ(pszDetailedFormat, "JPEG");
    3775           1 :             VSIFree(pszDetailedFormat);
    3776             :         }
    3777             :     }
    3778             : 
    3779             :     {
    3780           1 :         const GByte abyCanary[] = {0xDE, 0xAD, 0xBE, 0xEF};
    3781           1 :         std::vector<GByte> abyBuffer(nNeededSize + sizeof(abyCanary));
    3782           1 :         memcpy(&abyBuffer[nNeededSize], abyCanary, sizeof(abyCanary));
    3783           1 :         void *pabyBuffer = abyBuffer.data();
    3784           1 :         void **ppabyBuffer = &pabyBuffer;
    3785           1 :         size_t nProvidedSize = nNeededSize;
    3786           1 :         char *pszDetailedFormat = nullptr;
    3787           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3788             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20,
    3789             :                       20, 1, nullptr, ppabyBuffer, &nProvidedSize,
    3790             :                       &pszDetailedFormat),
    3791             :                   CE_None);
    3792           1 :         ASSERT_EQ(nProvidedSize, nNeededSize);
    3793           1 :         ASSERT_TRUE(*ppabyBuffer == pabyBuffer);
    3794           1 :         EXPECT_TRUE(pszDetailedFormat != nullptr);
    3795           1 :         if (pszDetailedFormat)
    3796             :         {
    3797           1 :             ASSERT_STREQ(pszDetailedFormat,
    3798             :                          "JPEG;frame_type=SOF0_baseline;bit_depth=8;num_"
    3799             :                          "components=1;colorspace=unknown");
    3800           1 :             VSIFree(pszDetailedFormat);
    3801             :         }
    3802           1 :         EXPECT_TRUE(
    3803             :             memcmp(&abyBuffer[nNeededSize], abyCanary, sizeof(abyCanary)) == 0);
    3804           1 :         EXPECT_EQ(abyBuffer[0], 0xFF);
    3805           1 :         EXPECT_EQ(abyBuffer[1], 0xD8);
    3806           1 :         EXPECT_EQ(abyBuffer[nNeededSize - 2], 0xFF);
    3807           1 :         EXPECT_EQ(abyBuffer[nNeededSize - 1], 0xD9);
    3808             : 
    3809             :         // Buffer larger than needed: OK
    3810           1 :         nProvidedSize = nNeededSize + 1;
    3811           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3812             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20,
    3813             :                       20, 1, nullptr, ppabyBuffer, &nProvidedSize, nullptr),
    3814             :                   CE_None);
    3815             : 
    3816             :         // Too small buffer
    3817           1 :         nProvidedSize = nNeededSize - 1;
    3818           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3819             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20,
    3820             :                       20, 1, nullptr, ppabyBuffer, &nProvidedSize, nullptr),
    3821             :                   CE_Failure);
    3822             : 
    3823             :         // Missing pointer to size
    3824           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3825             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20,
    3826             :                       20, 1, nullptr, ppabyBuffer, nullptr, nullptr),
    3827             :                   CE_Failure);
    3828             :     }
    3829             : 
    3830             :     // Let GDAL allocate buffer
    3831             :     {
    3832           1 :         void *pBuffer = nullptr;
    3833           1 :         size_t nGotSize = 0;
    3834           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3835             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 20,
    3836             :                       20, 1, nullptr, &pBuffer, &nGotSize, nullptr),
    3837             :                   CE_None);
    3838           1 :         EXPECT_EQ(nGotSize, nNeededSize);
    3839           1 :         EXPECT_NE(pBuffer, nullptr);
    3840           1 :         if (pBuffer != nullptr && nGotSize == nNeededSize && nNeededSize >= 2)
    3841             :         {
    3842           1 :             const GByte *pabyBuffer = static_cast<GByte *>(pBuffer);
    3843           1 :             EXPECT_EQ(pabyBuffer[0], 0xFF);
    3844           1 :             EXPECT_EQ(pabyBuffer[1], 0xD8);
    3845           1 :             EXPECT_EQ(pabyBuffer[nNeededSize - 2], 0xFF);
    3846           1 :             EXPECT_EQ(pabyBuffer[nNeededSize - 1], 0xD9);
    3847             :         }
    3848           1 :         VSIFree(pBuffer);
    3849             :     }
    3850             : 
    3851             :     // Cannot subset just one pixel
    3852           1 :     EXPECT_EQ(GDALDatasetReadCompressedData(
    3853             :                   GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 1, 1, 1,
    3854             :                   nullptr, nullptr, nullptr, nullptr),
    3855             :               CE_Failure);
    3856             : 
    3857           1 :     EXPECT_EQ(GDALDatasetReadCompressedData(
    3858             :                   GDALDataset::ToHandle(poSrcDS.get()), "wrong_format", 0, 0,
    3859             :                   20, 20, 1, nullptr, nullptr, nullptr, nullptr),
    3860             :               CE_Failure);
    3861             : }
    3862             : 
    3863             : // Test GDALDataset::GetCompressionFormats() and ReadCompressedData()
    3864           4 : TEST_F(test_gdal, gtiff_ReadCompressedData_jpeg_rgba)
    3865             : {
    3866           1 :     if (!GDALGetDriverByName("GTiff"))
    3867             :     {
    3868           0 :         GTEST_SKIP() << "GTiff driver missing";
    3869             :     }
    3870           1 :     if (GDALGetDriverByName("JPEG") == nullptr)
    3871             :     {
    3872           0 :         GTEST_SKIP() << "JPEG support missing";
    3873             :     }
    3874             : 
    3875             :     GDALDatasetUniquePtr poSrcDS(GDALDataset::FromHandle(
    3876           1 :         GDALDataset::Open((tut::common::data_basedir +
    3877             :                            "/../../gcore/data/stefan_full_rgba_jpeg_contig.tif")
    3878           1 :                               .c_str())));
    3879           1 :     ASSERT_TRUE(poSrcDS);
    3880             : 
    3881             :     const CPLStringList aosRet(GDALDatasetGetCompressionFormats(
    3882           1 :         GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 162, 16, 4, nullptr));
    3883           1 :     EXPECT_EQ(aosRet.size(), 1);
    3884           1 :     if (aosRet.size() == 1)
    3885             :     {
    3886           1 :         EXPECT_STREQ(aosRet[0], "JPEG;colorspace=RGBA");
    3887             :     }
    3888             : 
    3889             :     // Let GDAL allocate buffer
    3890             :     {
    3891           1 :         void *pBuffer = nullptr;
    3892           1 :         size_t nGotSize = 0;
    3893           1 :         char *pszDetailedFormat = nullptr;
    3894           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3895             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 162,
    3896             :                       16, 4, nullptr, &pBuffer, &nGotSize, &pszDetailedFormat),
    3897             :                   CE_None);
    3898           1 :         if (pszDetailedFormat)
    3899             :         {
    3900           1 :             ASSERT_STREQ(pszDetailedFormat,
    3901             :                          "JPEG;frame_type=SOF0_baseline;bit_depth=8;num_"
    3902             :                          "components=4;colorspace=RGBA");
    3903           1 :             VSIFree(pszDetailedFormat);
    3904             :         }
    3905           1 :         VSIFree(pBuffer);
    3906             :     }
    3907             : }
    3908             : 
    3909             : // Test GDALDataset::GetCompressionFormats() and ReadCompressedData()
    3910           4 : TEST_F(test_gdal, jpeg_ReadCompressedData)
    3911             : {
    3912           1 :     if (GDALGetDriverByName("JPEG") == nullptr)
    3913             :     {
    3914           0 :         GTEST_SKIP() << "JPEG support missing";
    3915             :     }
    3916             : 
    3917           1 :     GDALDatasetUniquePtr poSrcDS(GDALDataset::FromHandle(GDALDataset::Open(
    3918           1 :         (tut::common::data_basedir + "/../../gdrivers/data/jpeg/albania.jpg")
    3919           1 :             .c_str())));
    3920           1 :     ASSERT_TRUE(poSrcDS);
    3921             : 
    3922             :     const CPLStringList aosRet(GDALDatasetGetCompressionFormats(
    3923           1 :         GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 361, 260, 3, nullptr));
    3924           1 :     EXPECT_EQ(aosRet.size(), 1);
    3925           1 :     if (aosRet.size() == 1)
    3926             :     {
    3927           1 :         EXPECT_STREQ(aosRet[0],
    3928             :                      "JPEG;frame_type=SOF0_baseline;bit_depth=8;num_components="
    3929             :                      "3;subsampling=4:2:0;colorspace=YCbCr");
    3930             :     }
    3931             : 
    3932             :     size_t nUpperBoundSize;
    3933           1 :     EXPECT_EQ(GDALDatasetReadCompressedData(
    3934             :                   GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 361, 260,
    3935             :                   3, nullptr, nullptr, &nUpperBoundSize, nullptr),
    3936             :               CE_None);
    3937           1 :     EXPECT_EQ(nUpperBoundSize, 12574);
    3938             : 
    3939             :     {
    3940           1 :         std::vector<GByte> abyBuffer(nUpperBoundSize);
    3941           1 :         void *pabyBuffer = abyBuffer.data();
    3942           1 :         void **ppabyBuffer = &pabyBuffer;
    3943           1 :         size_t nSize = nUpperBoundSize;
    3944           1 :         char *pszDetailedFormat = nullptr;
    3945           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3946             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 361,
    3947             :                       260, 3, nullptr, ppabyBuffer, &nSize, &pszDetailedFormat),
    3948             :                   CE_None);
    3949           1 :         ASSERT_LT(nSize, nUpperBoundSize);
    3950           1 :         ASSERT_TRUE(*ppabyBuffer == pabyBuffer);
    3951           1 :         EXPECT_TRUE(pszDetailedFormat != nullptr);
    3952           1 :         if (pszDetailedFormat)
    3953             :         {
    3954           1 :             ASSERT_STREQ(pszDetailedFormat,
    3955             :                          "JPEG;frame_type=SOF0_baseline;bit_depth=8;num_"
    3956             :                          "components=3;subsampling=4:2:0;colorspace=YCbCr");
    3957           1 :             VSIFree(pszDetailedFormat);
    3958             :         }
    3959           1 :         EXPECT_EQ(abyBuffer[0], 0xFF);
    3960           1 :         EXPECT_EQ(abyBuffer[1], 0xD8);
    3961           1 :         EXPECT_EQ(abyBuffer[nSize - 2], 0xFF);
    3962           1 :         EXPECT_EQ(abyBuffer[nSize - 1], 0xD9);
    3963             : 
    3964             :         // Buffer larger than needed: OK
    3965           1 :         nSize = nUpperBoundSize + 1;
    3966           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3967             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 361,
    3968             :                       260, 3, nullptr, ppabyBuffer, &nSize, nullptr),
    3969             :                   CE_None);
    3970             : 
    3971             :         // Too small buffer
    3972           1 :         nSize = nUpperBoundSize - 1;
    3973           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3974             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 361,
    3975             :                       260, 3, nullptr, ppabyBuffer, &nSize, nullptr),
    3976             :                   CE_Failure);
    3977             : 
    3978             :         // Missing pointer to size
    3979           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3980             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 361,
    3981             :                       260, 3, nullptr, ppabyBuffer, nullptr, nullptr),
    3982             :                   CE_Failure);
    3983             :     }
    3984             : 
    3985             :     // Let GDAL allocate buffer
    3986             :     {
    3987           1 :         void *pBuffer = nullptr;
    3988           1 :         size_t nSize = nUpperBoundSize;
    3989           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    3990             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 361,
    3991             :                       260, 3, nullptr, &pBuffer, &nSize, nullptr),
    3992             :                   CE_None);
    3993           1 :         EXPECT_GT(nSize, 4);
    3994           1 :         EXPECT_LT(nSize, nUpperBoundSize);
    3995           1 :         EXPECT_NE(pBuffer, nullptr);
    3996           1 :         if (pBuffer != nullptr && nSize >= 4 && nSize <= nUpperBoundSize)
    3997             :         {
    3998           1 :             const GByte *pabyBuffer = static_cast<GByte *>(pBuffer);
    3999           1 :             EXPECT_EQ(pabyBuffer[0], 0xFF);
    4000           1 :             EXPECT_EQ(pabyBuffer[1], 0xD8);
    4001           1 :             EXPECT_EQ(pabyBuffer[nSize - 2], 0xFF);
    4002           1 :             EXPECT_EQ(pabyBuffer[nSize - 1], 0xD9);
    4003             :         }
    4004           1 :         VSIFree(pBuffer);
    4005             :     }
    4006             : }
    4007             : 
    4008             : // Test GDALDataset::GetCompressionFormats() and ReadCompressedData()
    4009           4 : TEST_F(test_gdal, jpegxl_ReadCompressedData)
    4010             : {
    4011           1 :     if (GDALGetDriverByName("JPEGXL") == nullptr)
    4012             :     {
    4013           0 :         GTEST_SKIP() << "JPEGXL support missing";
    4014             :     }
    4015             : 
    4016           1 :     GDALDatasetUniquePtr poSrcDS(GDALDataset::FromHandle(GDALDataset::Open(
    4017           1 :         (tut::common::data_basedir + "/../../gdrivers/data/jpegxl/byte.jxl")
    4018           1 :             .c_str())));
    4019           1 :     ASSERT_TRUE(poSrcDS);
    4020             : 
    4021             :     const CPLStringList aosRet(GDALDatasetGetCompressionFormats(
    4022           1 :         GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 20, 20, 1, nullptr));
    4023           1 :     EXPECT_EQ(aosRet.size(), 1);
    4024           1 :     if (aosRet.size() == 1)
    4025             :     {
    4026           1 :         EXPECT_STREQ(aosRet[0], "JXL");
    4027             :     }
    4028             : 
    4029             :     size_t nUpperBoundSize;
    4030           1 :     EXPECT_EQ(GDALDatasetReadCompressedData(
    4031             :                   GDALDataset::ToHandle(poSrcDS.get()), "JXL", 0, 0, 20, 20, 1,
    4032             :                   nullptr, nullptr, &nUpperBoundSize, nullptr),
    4033             :               CE_None);
    4034           1 :     EXPECT_EQ(nUpperBoundSize, 719);
    4035             : 
    4036             :     {
    4037           1 :         std::vector<GByte> abyBuffer(nUpperBoundSize);
    4038           1 :         void *pabyBuffer = abyBuffer.data();
    4039           1 :         void **ppabyBuffer = &pabyBuffer;
    4040           1 :         size_t nSize = nUpperBoundSize;
    4041           1 :         char *pszDetailedFormat = nullptr;
    4042           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4043             :                       GDALDataset::ToHandle(poSrcDS.get()), "JXL", 0, 0, 20, 20,
    4044             :                       1, nullptr, ppabyBuffer, &nSize, &pszDetailedFormat),
    4045             :                   CE_None);
    4046           1 :         ASSERT_LT(nSize, nUpperBoundSize);
    4047           1 :         ASSERT_TRUE(*ppabyBuffer == pabyBuffer);
    4048           1 :         EXPECT_TRUE(pszDetailedFormat != nullptr);
    4049           1 :         if (pszDetailedFormat)
    4050             :         {
    4051           1 :             ASSERT_STREQ(pszDetailedFormat, "JXL");
    4052           1 :             VSIFree(pszDetailedFormat);
    4053             :         }
    4054           1 :         EXPECT_EQ(abyBuffer[0], 0x00);
    4055           1 :         EXPECT_EQ(abyBuffer[1], 0x00);
    4056           1 :         EXPECT_EQ(abyBuffer[2], 0x00);
    4057           1 :         EXPECT_EQ(abyBuffer[3], 0x0C);
    4058           1 :         EXPECT_EQ(abyBuffer[nSize - 2], 0x4C);
    4059           1 :         EXPECT_EQ(abyBuffer[nSize - 1], 0x01);
    4060             : 
    4061             :         // Buffer larger than needed: OK
    4062           1 :         nSize = nUpperBoundSize + 1;
    4063           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4064             :                       GDALDataset::ToHandle(poSrcDS.get()), "JXL", 0, 0, 20, 20,
    4065             :                       1, nullptr, ppabyBuffer, &nSize, nullptr),
    4066             :                   CE_None);
    4067             : 
    4068             :         // Too small buffer
    4069           1 :         nSize = nUpperBoundSize - 1;
    4070           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4071             :                       GDALDataset::ToHandle(poSrcDS.get()), "JXL", 0, 0, 20, 20,
    4072             :                       1, nullptr, ppabyBuffer, &nSize, nullptr),
    4073             :                   CE_Failure);
    4074             : 
    4075             :         // Missing pointer to size
    4076           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4077             :                       GDALDataset::ToHandle(poSrcDS.get()), "JXL", 0, 0, 20, 20,
    4078             :                       1, nullptr, ppabyBuffer, nullptr, nullptr),
    4079             :                   CE_Failure);
    4080             :     }
    4081             : 
    4082             :     // Let GDAL allocate buffer
    4083             :     {
    4084           1 :         void *pBuffer = nullptr;
    4085           1 :         size_t nSize = nUpperBoundSize;
    4086           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4087             :                       GDALDataset::ToHandle(poSrcDS.get()), "JXL", 0, 0, 20, 20,
    4088             :                       1, nullptr, &pBuffer, &nSize, nullptr),
    4089             :                   CE_None);
    4090           1 :         EXPECT_GT(nSize, 6);
    4091           1 :         EXPECT_LT(nSize, nUpperBoundSize);
    4092           1 :         EXPECT_NE(pBuffer, nullptr);
    4093           1 :         if (pBuffer != nullptr && nSize >= 6 && nSize <= nUpperBoundSize)
    4094             :         {
    4095           1 :             const GByte *pabyBuffer = static_cast<GByte *>(pBuffer);
    4096           1 :             EXPECT_EQ(pabyBuffer[0], 0x00);
    4097           1 :             EXPECT_EQ(pabyBuffer[1], 0x00);
    4098           1 :             EXPECT_EQ(pabyBuffer[2], 0x00);
    4099           1 :             EXPECT_EQ(pabyBuffer[3], 0x0C);
    4100           1 :             EXPECT_EQ(pabyBuffer[nSize - 2], 0x4C);
    4101           1 :             EXPECT_EQ(pabyBuffer[nSize - 1], 0x01);
    4102             :         }
    4103           1 :         VSIFree(pBuffer);
    4104             :     }
    4105             : }
    4106             : 
    4107             : // Test GDALDataset::GetCompressionFormats() and ReadCompressedData()
    4108           4 : TEST_F(test_gdal, jpegxl_jpeg_compatible_ReadCompressedData)
    4109             : {
    4110           1 :     auto poDrv = GDALDriver::FromHandle(GDALGetDriverByName("JPEGXL"));
    4111           1 :     if (poDrv == nullptr)
    4112             :     {
    4113           0 :         GTEST_SKIP() << "JPEGXL support missing";
    4114             :     }
    4115             : 
    4116           1 :     GDALDatasetUniquePtr poSrcDS(GDALDataset::FromHandle(GDALDataset::Open(
    4117           1 :         (tut::common::data_basedir +
    4118             :          "/../../gdrivers/data/jpegxl/exif_orientation/F1.jxl")
    4119           1 :             .c_str())));
    4120           1 :     ASSERT_TRUE(poSrcDS);
    4121             : 
    4122             :     const CPLStringList aosRet(GDALDatasetGetCompressionFormats(
    4123           1 :         GDALDataset::ToHandle(poSrcDS.get()), 0, 0, 3, 5, 1, nullptr));
    4124           1 :     EXPECT_EQ(aosRet.size(), 2);
    4125           1 :     if (aosRet.size() == 2)
    4126             :     {
    4127           1 :         EXPECT_STREQ(aosRet[0], "JXL");
    4128           1 :         EXPECT_STREQ(aosRet[1], "JPEG");
    4129             :     }
    4130             : 
    4131             :     size_t nUpperBoundSize;
    4132           1 :     EXPECT_EQ(GDALDatasetReadCompressedData(
    4133             :                   GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 3, 5, 1,
    4134             :                   nullptr, nullptr, &nUpperBoundSize, nullptr),
    4135             :               CE_None);
    4136           1 :     EXPECT_EQ(nUpperBoundSize, 235);
    4137             : 
    4138             :     {
    4139           1 :         std::vector<GByte> abyBuffer(nUpperBoundSize);
    4140           1 :         void *pabyBuffer = abyBuffer.data();
    4141           1 :         void **ppabyBuffer = &pabyBuffer;
    4142           1 :         size_t nSize = nUpperBoundSize;
    4143           1 :         char *pszDetailedFormat = nullptr;
    4144           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4145             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 3, 5,
    4146             :                       1, nullptr, ppabyBuffer, &nSize, &pszDetailedFormat),
    4147             :                   CE_None);
    4148           1 :         ASSERT_LE(nSize, nUpperBoundSize);
    4149           1 :         ASSERT_TRUE(*ppabyBuffer == pabyBuffer);
    4150           1 :         EXPECT_TRUE(pszDetailedFormat != nullptr);
    4151           1 :         if (pszDetailedFormat)
    4152             :         {
    4153           1 :             ASSERT_STREQ(pszDetailedFormat,
    4154             :                          "JPEG;frame_type=SOF0_baseline;bit_depth=8;num_"
    4155             :                          "components=1;colorspace=unknown");
    4156           1 :             VSIFree(pszDetailedFormat);
    4157             :         }
    4158           1 :         EXPECT_EQ(abyBuffer[0], 0xFF);
    4159           1 :         EXPECT_EQ(abyBuffer[1], 0xD8);
    4160           1 :         EXPECT_EQ(abyBuffer[nSize - 2], 0xFF);
    4161           1 :         EXPECT_EQ(abyBuffer[nSize - 1], 0xD9);
    4162             : 
    4163             :         // Buffer larger than needed: OK
    4164           1 :         nSize = nUpperBoundSize + 1;
    4165           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4166             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 3, 5,
    4167             :                       1, nullptr, ppabyBuffer, &nSize, nullptr),
    4168             :                   CE_None);
    4169             : 
    4170             :         // Too small buffer
    4171           1 :         nSize = nUpperBoundSize - 1;
    4172           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4173             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 3, 5,
    4174             :                       1, nullptr, ppabyBuffer, &nSize, nullptr),
    4175             :                   CE_Failure);
    4176             : 
    4177             :         // Missing pointer to size
    4178           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4179             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 3, 5,
    4180             :                       1, nullptr, ppabyBuffer, nullptr, nullptr),
    4181             :                   CE_Failure);
    4182             :     }
    4183             : 
    4184             :     // Let GDAL allocate buffer
    4185             :     {
    4186           1 :         void *pBuffer = nullptr;
    4187           1 :         size_t nSize = nUpperBoundSize;
    4188           1 :         EXPECT_EQ(GDALDatasetReadCompressedData(
    4189             :                       GDALDataset::ToHandle(poSrcDS.get()), "JPEG", 0, 0, 3, 5,
    4190             :                       1, nullptr, &pBuffer, &nSize, nullptr),
    4191             :                   CE_None);
    4192           1 :         EXPECT_GT(nSize, 4);
    4193           1 :         EXPECT_LE(nSize, nUpperBoundSize);
    4194           1 :         EXPECT_NE(pBuffer, nullptr);
    4195           1 :         if (pBuffer != nullptr && nSize >= 4 && nSize <= nUpperBoundSize)
    4196             :         {
    4197           1 :             const GByte *pabyBuffer = static_cast<GByte *>(pBuffer);
    4198           1 :             EXPECT_EQ(pabyBuffer[0], 0xFF);
    4199           1 :             EXPECT_EQ(pabyBuffer[1], 0xD8);
    4200           1 :             EXPECT_EQ(pabyBuffer[nSize - 2], 0xFF);
    4201           1 :             EXPECT_EQ(pabyBuffer[nSize - 1], 0xD9);
    4202             :         }
    4203           1 :         VSIFree(pBuffer);
    4204             :     }
    4205             : }
    4206             : 
    4207             : // Test GDAL_OF_SHARED flag and open options
    4208           4 : TEST_F(test_gdal, open_shared_open_options)
    4209             : {
    4210           1 :     if (!GDALGetDriverByName("GTiff"))
    4211             :     {
    4212           0 :         GTEST_SKIP() << "GTiff driver missing";
    4213             :     }
    4214             : 
    4215           1 :     CPLErrorReset();
    4216           1 :     const char *const apszOpenOptions[] = {"OVERVIEW_LEVEL=NONE", nullptr};
    4217             :     {
    4218             :         GDALDataset *poDS1 =
    4219           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4220           1 :                               nullptr, apszOpenOptions);
    4221             :         GDALDataset *poDS2 =
    4222           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4223           1 :                               nullptr, apszOpenOptions);
    4224           1 :         EXPECT_EQ(CPLGetLastErrorType(), CE_None);
    4225           1 :         EXPECT_NE(poDS1, nullptr);
    4226           1 :         EXPECT_NE(poDS2, nullptr);
    4227           1 :         EXPECT_EQ(poDS1, poDS2);
    4228           1 :         GDALClose(poDS1);
    4229           1 :         GDALClose(poDS2);
    4230             :     }
    4231             :     {
    4232             :         GDALDataset *poDS1 =
    4233           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4234           1 :                               nullptr, apszOpenOptions);
    4235           1 :         GDALDataset *poDS2 = GDALDataset::Open(
    4236           1 :             GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED, nullptr, nullptr);
    4237             :         GDALDataset *poDS3 =
    4238           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4239           1 :                               nullptr, apszOpenOptions);
    4240           1 :         EXPECT_EQ(CPLGetLastErrorType(), CE_None);
    4241           1 :         EXPECT_NE(poDS1, nullptr);
    4242           1 :         EXPECT_NE(poDS2, nullptr);
    4243           1 :         EXPECT_NE(poDS3, nullptr);
    4244           1 :         EXPECT_NE(poDS1, poDS2);
    4245           1 :         EXPECT_EQ(poDS1, poDS3);
    4246           1 :         GDALClose(poDS1);
    4247           1 :         GDALClose(poDS2);
    4248           1 :         GDALClose(poDS3);
    4249             :     }
    4250             :     {
    4251           1 :         GDALDataset *poDS1 = GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif",
    4252             :                                                GDAL_OF_SHARED | GDAL_OF_UPDATE,
    4253           1 :                                                nullptr, apszOpenOptions);
    4254             :         // We allow to re-use a shared dataset in update mode when requesting it in read-only
    4255             :         GDALDataset *poDS2 =
    4256           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4257           1 :                               nullptr, apszOpenOptions);
    4258           1 :         EXPECT_EQ(CPLGetLastErrorType(), CE_None);
    4259           1 :         EXPECT_NE(poDS1, nullptr);
    4260           1 :         EXPECT_NE(poDS2, nullptr);
    4261           1 :         EXPECT_EQ(poDS1, poDS2);
    4262           1 :         GDALClose(poDS1);
    4263           1 :         GDALClose(poDS2);
    4264             :     }
    4265             :     {
    4266           1 :         GDALDataset *poDS1 = GDALDataset::Open(
    4267           1 :             GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED, nullptr, nullptr);
    4268             :         GDALDataset *poDS2 =
    4269           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4270           1 :                               nullptr, apszOpenOptions);
    4271             :         GDALDataset *poDS3 =
    4272           1 :             GDALDataset::Open(GCORE_DATA_DIR "rgbsmall.tif", GDAL_OF_SHARED,
    4273           1 :                               nullptr, apszOpenOptions);
    4274           1 :         EXPECT_EQ(CPLGetLastErrorType(), CE_None);
    4275           1 :         EXPECT_NE(poDS1, nullptr);
    4276           1 :         EXPECT_NE(poDS2, nullptr);
    4277           1 :         EXPECT_NE(poDS3, nullptr);
    4278           1 :         EXPECT_NE(poDS1, poDS2);
    4279           1 :         EXPECT_EQ(poDS2, poDS3);
    4280           1 :         GDALClose(poDS1);
    4281           1 :         GDALClose(poDS2);
    4282           1 :         GDALClose(poDS3);
    4283             :     }
    4284             : }
    4285             : 
    4286             : // Test DropCache() to check that no data is saved on disk
    4287           4 : TEST_F(test_gdal, drop_cache)
    4288             : {
    4289           1 :     CPLErrorReset();
    4290             :     {
    4291           1 :         GDALDriverManager *gdalDriverManager = GetGDALDriverManager();
    4292           1 :         if (!gdalDriverManager)
    4293           0 :             return;
    4294           1 :         GDALDriver *enviDriver = gdalDriverManager->GetDriverByName("ENVI");
    4295           1 :         if (!enviDriver)
    4296           0 :             return;
    4297           1 :         const char *enviOptions[] = {"SUFFIX=ADD", "INTERLEAVE=BIL", nullptr};
    4298             : 
    4299           1 :         const char *filename = GCORE_DATA_DIR "test_drop_cache.bil";
    4300             : 
    4301             :         auto poDS = std::unique_ptr<GDALDataset>(enviDriver->Create(
    4302           1 :             filename, 1, 1, 1, GDALDataType::GDT_Float32, enviOptions));
    4303           1 :         if (!poDS)
    4304           0 :             return;
    4305           1 :         poDS->GetRasterBand(1)->Fill(1);
    4306           1 :         poDS->DropCache();
    4307           1 :         poDS.reset();
    4308             : 
    4309           1 :         poDS.reset(
    4310             :             GDALDataset::Open(filename, GDAL_OF_SHARED, nullptr, nullptr));
    4311           1 :         if (!poDS)
    4312           0 :             return;
    4313             : 
    4314           1 :         EXPECT_EQ(GDALChecksumImage(poDS->GetRasterBand(1), 0, 0, 1, 1), 0);
    4315           1 :         poDS->MarkSuppressOnClose();
    4316           1 :         poDS.reset();
    4317             :     }
    4318             : }
    4319             : 
    4320             : // Test gdal::gcp class
    4321           4 : TEST_F(test_gdal, gdal_gcp_class)
    4322             : {
    4323             :     {
    4324           2 :         gdal::GCP gcp;
    4325           1 :         EXPECT_STREQ(gcp.Id(), "");
    4326           1 :         EXPECT_STREQ(gcp.Info(), "");
    4327           1 :         EXPECT_EQ(gcp.Pixel(), 0.0);
    4328           1 :         EXPECT_EQ(gcp.Line(), 0.0);
    4329           1 :         EXPECT_EQ(gcp.X(), 0.0);
    4330           1 :         EXPECT_EQ(gcp.Y(), 0.0);
    4331           1 :         EXPECT_EQ(gcp.Z(), 0.0);
    4332             :     }
    4333             :     {
    4334           2 :         gdal::GCP gcp("id", "info", 1.5, 2.5, 3.5, 4.5, 5.5);
    4335           1 :         EXPECT_STREQ(gcp.Id(), "id");
    4336           1 :         EXPECT_STREQ(gcp.Info(), "info");
    4337           1 :         EXPECT_EQ(gcp.Pixel(), 1.5);
    4338           1 :         EXPECT_EQ(gcp.Line(), 2.5);
    4339           1 :         EXPECT_EQ(gcp.X(), 3.5);
    4340           1 :         EXPECT_EQ(gcp.Y(), 4.5);
    4341           1 :         EXPECT_EQ(gcp.Z(), 5.5);
    4342             : 
    4343           1 :         gcp.SetId("id2");
    4344           1 :         gcp.SetInfo("info2");
    4345           1 :         gcp.Pixel() = -1.5;
    4346           1 :         gcp.Line() = -2.5;
    4347           1 :         gcp.X() = -3.5;
    4348           1 :         gcp.Y() = -4.5;
    4349           1 :         gcp.Z() = -5.5;
    4350           1 :         EXPECT_STREQ(gcp.Id(), "id2");
    4351           1 :         EXPECT_STREQ(gcp.Info(), "info2");
    4352           1 :         EXPECT_EQ(gcp.Pixel(), -1.5);
    4353           1 :         EXPECT_EQ(gcp.Line(), -2.5);
    4354           1 :         EXPECT_EQ(gcp.X(), -3.5);
    4355           1 :         EXPECT_EQ(gcp.Y(), -4.5);
    4356           1 :         EXPECT_EQ(gcp.Z(), -5.5);
    4357             : 
    4358             :         {
    4359           2 :             gdal::GCP gcp_copy(gcp);
    4360           1 :             EXPECT_STREQ(gcp_copy.Id(), "id2");
    4361           1 :             EXPECT_STREQ(gcp_copy.Info(), "info2");
    4362           1 :             EXPECT_EQ(gcp_copy.Pixel(), -1.5);
    4363           1 :             EXPECT_EQ(gcp_copy.Line(), -2.5);
    4364           1 :             EXPECT_EQ(gcp_copy.X(), -3.5);
    4365           1 :             EXPECT_EQ(gcp_copy.Y(), -4.5);
    4366           1 :             EXPECT_EQ(gcp_copy.Z(), -5.5);
    4367             :         }
    4368             : 
    4369             :         {
    4370           2 :             gdal::GCP gcp_copy;
    4371           1 :             gcp_copy = gcp;
    4372           1 :             EXPECT_STREQ(gcp_copy.Id(), "id2");
    4373           1 :             EXPECT_STREQ(gcp_copy.Info(), "info2");
    4374           1 :             EXPECT_EQ(gcp_copy.Pixel(), -1.5);
    4375           1 :             EXPECT_EQ(gcp_copy.Line(), -2.5);
    4376           1 :             EXPECT_EQ(gcp_copy.X(), -3.5);
    4377           1 :             EXPECT_EQ(gcp_copy.Y(), -4.5);
    4378           1 :             EXPECT_EQ(gcp_copy.Z(), -5.5);
    4379             :         }
    4380             : 
    4381             :         {
    4382           2 :             gdal::GCP gcp_copy(gcp);
    4383           2 :             gdal::GCP gcp_from_moved(std::move(gcp_copy));
    4384           1 :             EXPECT_STREQ(gcp_from_moved.Id(), "id2");
    4385           1 :             EXPECT_STREQ(gcp_from_moved.Info(), "info2");
    4386           1 :             EXPECT_EQ(gcp_from_moved.Pixel(), -1.5);
    4387           1 :             EXPECT_EQ(gcp_from_moved.Line(), -2.5);
    4388           1 :             EXPECT_EQ(gcp_from_moved.X(), -3.5);
    4389           1 :             EXPECT_EQ(gcp_from_moved.Y(), -4.5);
    4390           1 :             EXPECT_EQ(gcp_from_moved.Z(), -5.5);
    4391             :         }
    4392             : 
    4393             :         {
    4394           2 :             gdal::GCP gcp_copy(gcp);
    4395           2 :             gdal::GCP gcp_from_moved;
    4396           1 :             gcp_from_moved = std::move(gcp_copy);
    4397           1 :             EXPECT_STREQ(gcp_from_moved.Id(), "id2");
    4398           1 :             EXPECT_STREQ(gcp_from_moved.Info(), "info2");
    4399           1 :             EXPECT_EQ(gcp_from_moved.Pixel(), -1.5);
    4400           1 :             EXPECT_EQ(gcp_from_moved.Line(), -2.5);
    4401           1 :             EXPECT_EQ(gcp_from_moved.X(), -3.5);
    4402           1 :             EXPECT_EQ(gcp_from_moved.Y(), -4.5);
    4403           1 :             EXPECT_EQ(gcp_from_moved.Z(), -5.5);
    4404             :         }
    4405             : 
    4406             :         {
    4407           1 :             const GDAL_GCP *c_gcp = gcp.c_ptr();
    4408           1 :             EXPECT_STREQ(c_gcp->pszId, "id2");
    4409           1 :             EXPECT_STREQ(c_gcp->pszInfo, "info2");
    4410           1 :             EXPECT_EQ(c_gcp->dfGCPPixel, -1.5);
    4411           1 :             EXPECT_EQ(c_gcp->dfGCPLine, -2.5);
    4412           1 :             EXPECT_EQ(c_gcp->dfGCPX, -3.5);
    4413           1 :             EXPECT_EQ(c_gcp->dfGCPY, -4.5);
    4414           1 :             EXPECT_EQ(c_gcp->dfGCPZ, -5.5);
    4415             : 
    4416           2 :             const gdal::GCP gcp_from_c(*c_gcp);
    4417           1 :             EXPECT_STREQ(gcp_from_c.Id(), "id2");
    4418           1 :             EXPECT_STREQ(gcp_from_c.Info(), "info2");
    4419           1 :             EXPECT_EQ(gcp_from_c.Pixel(), -1.5);
    4420           1 :             EXPECT_EQ(gcp_from_c.Line(), -2.5);
    4421           1 :             EXPECT_EQ(gcp_from_c.X(), -3.5);
    4422           1 :             EXPECT_EQ(gcp_from_c.Y(), -4.5);
    4423           1 :             EXPECT_EQ(gcp_from_c.Z(), -5.5);
    4424             :         }
    4425             :     }
    4426             : 
    4427             :     {
    4428             :         const std::vector<gdal::GCP> gcps{
    4429             :             gdal::GCP{nullptr, nullptr, 0, 0, 0, 0, 0},
    4430           4 :             gdal::GCP{"id", "info", 1.5, 2.5, 3.5, 4.5, 5.5}};
    4431             : 
    4432           1 :         const GDAL_GCP *c_gcps = gdal::GCP::c_ptr(gcps);
    4433           1 :         EXPECT_STREQ(c_gcps[1].pszId, "id");
    4434           1 :         EXPECT_STREQ(c_gcps[1].pszInfo, "info");
    4435           1 :         EXPECT_EQ(c_gcps[1].dfGCPPixel, 1.5);
    4436           1 :         EXPECT_EQ(c_gcps[1].dfGCPLine, 2.5);
    4437           1 :         EXPECT_EQ(c_gcps[1].dfGCPX, 3.5);
    4438           1 :         EXPECT_EQ(c_gcps[1].dfGCPY, 4.5);
    4439           1 :         EXPECT_EQ(c_gcps[1].dfGCPZ, 5.5);
    4440             : 
    4441             :         const auto gcps_from_c =
    4442           1 :             gdal::GCP::fromC(c_gcps, static_cast<int>(gcps.size()));
    4443           1 :         ASSERT_EQ(gcps_from_c.size(), gcps.size());
    4444           3 :         for (size_t i = 0; i < gcps.size(); ++i)
    4445             :         {
    4446           2 :             EXPECT_STREQ(gcps_from_c[i].Id(), gcps[i].Id());
    4447           2 :             EXPECT_STREQ(gcps_from_c[i].Info(), gcps[i].Info());
    4448           2 :             EXPECT_EQ(gcps_from_c[i].Pixel(), gcps[i].Pixel());
    4449           2 :             EXPECT_EQ(gcps_from_c[i].Line(), gcps[i].Line());
    4450           2 :             EXPECT_EQ(gcps_from_c[i].X(), gcps[i].X());
    4451           2 :             EXPECT_EQ(gcps_from_c[i].Y(), gcps[i].Y());
    4452           2 :             EXPECT_EQ(gcps_from_c[i].Z(), gcps[i].Z());
    4453             :         }
    4454             :     }
    4455             : }
    4456             : 
    4457           4 : TEST_F(test_gdal, RasterIO_gdt_unknown)
    4458             : {
    4459             :     GDALDatasetUniquePtr poDS(GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    4460           2 :                                   ->Create("", 1, 1, 1, GDT_Float64, nullptr));
    4461           2 :     CPLErrorHandlerPusher oErrorHandler(CPLQuietErrorHandler);
    4462           1 :     GByte b = 0;
    4463             :     GDALRasterIOExtraArg sExtraArg;
    4464           1 :     INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    4465           1 :     EXPECT_EQ(poDS->RasterIO(GF_Read, 0, 0, 1, 1, &b, 1, 1, GDT_Unknown, 1,
    4466             :                              nullptr, 0, 0, 0, &sExtraArg),
    4467             :               CE_Failure);
    4468           1 :     EXPECT_EQ(poDS->RasterIO(GF_Read, 0, 0, 1, 1, &b, 1, 1, GDT_TypeCount, 1,
    4469             :                              nullptr, 0, 0, 0, &sExtraArg),
    4470             :               CE_Failure);
    4471           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->RasterIO(GF_Read, 0, 0, 1, 1, &b, 1, 1,
    4472             :                                                GDT_Unknown, 0, 0, &sExtraArg),
    4473             :               CE_Failure);
    4474           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->RasterIO(GF_Read, 0, 0, 1, 1, &b, 1, 1,
    4475             :                                                GDT_TypeCount, 0, 0, &sExtraArg),
    4476             :               CE_Failure);
    4477           1 : }
    4478             : 
    4479           4 : TEST_F(test_gdal, CopyWords_gdt_unknown)
    4480             : {
    4481           2 :     CPLErrorHandlerPusher oErrorHandler(CPLQuietErrorHandler);
    4482           1 :     GByte b = 0;
    4483           1 :     GByte b2 = 0;
    4484           1 :     CPLErrorReset();
    4485           1 :     GDALCopyWords(&b, GDT_Byte, 0, &b2, GDT_Unknown, 0, 1);
    4486           1 :     EXPECT_EQ(CPLGetLastErrorType(), CE_Failure);
    4487           1 :     CPLErrorReset();
    4488           1 :     GDALCopyWords(&b, GDT_Unknown, 0, &b2, GDT_Byte, 0, 1);
    4489           1 :     EXPECT_EQ(CPLGetLastErrorType(), CE_Failure);
    4490           1 : }
    4491             : 
    4492             : // Test GDALRasterBand::ReadRaster
    4493           4 : TEST_F(test_gdal, ReadRaster)
    4494             : {
    4495             :     GDALDatasetUniquePtr poDS(GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    4496           1 :                                   ->Create("", 2, 3, 1, GDT_Float64, nullptr));
    4497           1 :     std::array<double, 6> buffer = {
    4498             :         -1e300, -1,     //////////////////////////////////////////////
    4499             :         1,      128,    //////////////////////////////////////////////
    4500             :         32768,  1e300,  //////////////////////////////////////////////
    4501             :     };
    4502             :     GDALRasterIOExtraArg sExtraArg;
    4503           1 :     INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    4504           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->RasterIO(
    4505             :                   GF_Write, 0, 0, 2, 3, buffer.data(), 2, 3, GDT_Float64,
    4506             :                   sizeof(double), 2 * sizeof(double), &sExtraArg),
    4507             :               CE_None);
    4508             : 
    4509             :     {
    4510           2 :         std::vector<uint8_t> res;
    4511           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4512           2 :         const auto expected_res = std::vector<uint8_t>{0, 0, 1, 128, 255, 255};
    4513           1 :         EXPECT_EQ(res, expected_res);
    4514             : 
    4515           1 :         std::fill(res.begin(), res.end(), 0);
    4516           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res, 0, 0, 2, 3, 2, 3),
    4517             :                   CE_None);
    4518           1 :         EXPECT_EQ(res, expected_res);
    4519             : 
    4520           1 :         std::fill(res.begin(), res.end(), 0);
    4521           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res, 0, 0, 2, 3), CE_None);
    4522           1 :         EXPECT_EQ(res, expected_res);
    4523             : 
    4524             : #if __cplusplus >= 202002L
    4525             :         std::fill(res.begin(), res.end(), 0);
    4526             :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(std::span<uint8_t>(res)),
    4527             :                   CE_None);
    4528             :         EXPECT_EQ(res, expected_res);
    4529             : #endif
    4530             : 
    4531           1 :         std::fill(res.begin(), res.end(), 0);
    4532           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4533           1 :         EXPECT_EQ(res, expected_res);
    4534             : 
    4535           1 :         std::fill(res.begin(), res.end(), 0);
    4536           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data(), res.size()),
    4537             :                   CE_None);
    4538           1 :         EXPECT_EQ(res, expected_res);
    4539             : 
    4540           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    4541             :         // Too small buffer size
    4542           1 :         EXPECT_EQ(
    4543             :             poDS->GetRasterBand(1)->ReadRaster(res.data(), res.size() - 1),
    4544             :             CE_Failure);
    4545           1 :         CPLPopErrorHandler();
    4546             : 
    4547           1 :         std::fill(res.begin(), res.end(), 0);
    4548           1 :         EXPECT_EQ(
    4549             :             poDS->GetRasterBand(1)->ReadRaster(res.data(), 0, 0, 0, 2, 3, 2, 3),
    4550             :             CE_None);
    4551           1 :         EXPECT_EQ(res, expected_res);
    4552             : 
    4553           1 :         std::fill(res.begin(), res.end(), 0);
    4554           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data(), 0, 0, 0, 2, 3),
    4555             :                   CE_None);
    4556           1 :         EXPECT_EQ(res, expected_res);
    4557             :     }
    4558             : 
    4559             :     {
    4560           2 :         std::vector<double> res;
    4561           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    4562             :         // Too large nBufXSize
    4563           1 :         EXPECT_EQ(
    4564             :             poDS->GetRasterBand(1)->ReadRaster(res, 0, 0, 1, 1, UINT32_MAX, 1),
    4565             :             CE_Failure);
    4566             : 
    4567           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data(), UINT32_MAX, 0,
    4568             :                                                      0, 1, 1, UINT32_MAX, 1),
    4569             :                   CE_Failure);
    4570             : 
    4571             :         // Too large nBufYSize
    4572           1 :         EXPECT_EQ(
    4573             :             poDS->GetRasterBand(1)->ReadRaster(res, 0, 0, 1, 1, 1, UINT32_MAX),
    4574             :             CE_Failure);
    4575             : 
    4576           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data(), UINT32_MAX, 0,
    4577             :                                                      0, 1, 1, 1, UINT32_MAX),
    4578             :                   CE_Failure);
    4579             : 
    4580           1 :         CPLPopErrorHandler();
    4581             :     }
    4582             : 
    4583             :     {
    4584           2 :         std::vector<double> res;
    4585           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    4586             :         // Huge nBufXSize x nBufYSize
    4587           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res, 0, 0, 1, 1, INT32_MAX,
    4588             :                                                      INT32_MAX),
    4589             :                   CE_Failure);
    4590           1 :         CPLPopErrorHandler();
    4591             :     }
    4592             : 
    4593             :     {
    4594           2 :         std::vector<double> res;
    4595           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res, 1, 2, 1, 1), CE_None);
    4596           2 :         const auto expected_res = std::vector<double>{1e300};
    4597           1 :         EXPECT_EQ(res, expected_res);
    4598             :     }
    4599             : 
    4600             :     {
    4601           2 :         std::vector<double> res;
    4602           1 :         CPLPushErrorHandler(CPLQuietErrorHandler);
    4603           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res, 1.1, 2.1, 0.9, 0.9),
    4604             :                   CE_Failure);
    4605           1 :         CPLPopErrorHandler();
    4606             : 
    4607           1 :         EXPECT_EQ(
    4608             :             poDS->GetRasterBand(1)->ReadRaster(res, 1.1, 2.1, 0.9, 0.9, 1, 1),
    4609             :             CE_None);
    4610           2 :         const auto expected_res = std::vector<double>{1e300};
    4611           1 :         EXPECT_EQ(res, expected_res);
    4612             :     }
    4613             : 
    4614             :     {
    4615           1 :         std::vector<double> res;
    4616           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res, 0.4, 0.5, 1.4, 1.5, 1,
    4617             :                                                      1, GRIORA_Bilinear),
    4618             :                   CE_None);
    4619           1 :         ASSERT_EQ(res.size(), 1U);
    4620           1 :         const double expected_res = -8.64198e+298;
    4621           1 :         EXPECT_NEAR(res[0], expected_res, std::fabs(expected_res) * 1e-6);
    4622             :     }
    4623             : 
    4624             :     // Test int8_t
    4625             :     {
    4626           2 :         std::vector<int8_t> res;
    4627           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4628             :         const auto expected_res =
    4629           2 :             std::vector<int8_t>{-128, -1, 1, 127, 127, 127};
    4630           1 :         EXPECT_EQ(res, expected_res);
    4631             : 
    4632           1 :         std::fill(res.begin(), res.end(), 0);
    4633           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4634           1 :         EXPECT_EQ(res, expected_res);
    4635             :     }
    4636             : 
    4637             :     // Test uint16_t
    4638             :     {
    4639           2 :         std::vector<uint16_t> res;
    4640           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4641             :         const auto expected_res =
    4642           2 :             std::vector<uint16_t>{0, 0, 1, 128, 32768, 65535};
    4643           1 :         EXPECT_EQ(res, expected_res);
    4644             : 
    4645           1 :         std::fill(res.begin(), res.end(), 0);
    4646           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4647           1 :         EXPECT_EQ(res, expected_res);
    4648             : 
    4649           1 :         std::fill(res.begin(), res.end(), 0);
    4650           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data(), res.size()),
    4651             :                   CE_None);
    4652           1 :         EXPECT_EQ(res, expected_res);
    4653             :     }
    4654             : 
    4655             :     // Test int16_t
    4656             :     {
    4657           2 :         std::vector<int16_t> res;
    4658           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4659             :         const auto expected_res =
    4660           2 :             std::vector<int16_t>{-32768, -1, 1, 128, 32767, 32767};
    4661           1 :         EXPECT_EQ(res, expected_res);
    4662             : 
    4663           1 :         std::fill(res.begin(), res.end(), 0);
    4664           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4665           1 :         EXPECT_EQ(res, expected_res);
    4666             :     }
    4667             : 
    4668             : #if 0
    4669             :     // Not allowed by C++ standard
    4670             :     // Test complex<int16_t>
    4671             :     {
    4672             :         std::vector<std::complex<int16_t>> res;
    4673             :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4674             :         const auto expected_res = std::vector<std::complex<int16_t>>{
    4675             :             -32768, -1, 1, 128, 32767, 32767};
    4676             :         EXPECT_EQ(res, expected_res);
    4677             : 
    4678             :         std::fill(res.begin(), res.end(), 0);
    4679             :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4680             :         EXPECT_EQ(res, expected_res);
    4681             :     }
    4682             : #endif
    4683             : 
    4684             :     // Test uint32_t
    4685             :     {
    4686           2 :         std::vector<uint32_t> res;
    4687           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4688             :         const auto expected_res =
    4689           2 :             std::vector<uint32_t>{0, 0, 1, 128, 32768, UINT32_MAX};
    4690           1 :         EXPECT_EQ(res, expected_res);
    4691             : 
    4692           1 :         std::fill(res.begin(), res.end(), 0);
    4693           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4694           1 :         EXPECT_EQ(res, expected_res);
    4695             :     }
    4696             : 
    4697             :     // Test int32_t
    4698             :     {
    4699           2 :         std::vector<int32_t> res;
    4700           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4701             :         const auto expected_res =
    4702           2 :             std::vector<int32_t>{INT32_MIN, -1, 1, 128, 32768, INT32_MAX};
    4703           1 :         EXPECT_EQ(res, expected_res);
    4704             : 
    4705           1 :         std::fill(res.begin(), res.end(), 0);
    4706           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4707           1 :         EXPECT_EQ(res, expected_res);
    4708             :     }
    4709             : 
    4710             : #if 0
    4711             :     // Not allowed by C++ standard
    4712             :     // Test complex<int32_t>
    4713             :     {
    4714             :         std::vector<std::complex<int32_t>> res;
    4715             :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4716             :         const auto expected_res = std::vector<std::complex<int32_t>>{
    4717             :             INT32_MIN, -1, 1, 128, 32768, INT32_MAX};
    4718             :         EXPECT_EQ(res, expected_res);
    4719             : 
    4720             :         std::fill(res.begin(), res.end(), 0);
    4721             :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4722             :         EXPECT_EQ(res, expected_res);
    4723             :     }
    4724             : #endif
    4725             : 
    4726             :     // Test uint64_t
    4727             :     {
    4728           2 :         std::vector<uint64_t> res;
    4729           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4730             :         const auto expected_res =
    4731           2 :             std::vector<uint64_t>{0, 0, 1, 128, 32768, UINT64_MAX};
    4732           1 :         EXPECT_EQ(res, expected_res);
    4733             : 
    4734           1 :         std::fill(res.begin(), res.end(), 0);
    4735           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4736           1 :         EXPECT_EQ(res, expected_res);
    4737             :     }
    4738             : 
    4739             :     // Test int64_t
    4740             :     {
    4741           2 :         std::vector<int64_t> res;
    4742           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4743             :         const auto expected_res =
    4744           2 :             std::vector<int64_t>{INT64_MIN, -1, 1, 128, 32768, INT64_MAX};
    4745           1 :         EXPECT_EQ(res, expected_res);
    4746             : 
    4747           1 :         std::fill(res.begin(), res.end(), 0);
    4748           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4749           1 :         EXPECT_EQ(res, expected_res);
    4750             :     }
    4751             : 
    4752             :     // Test float
    4753             :     {
    4754           2 :         std::vector<float> res;
    4755           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4756             :         const auto expected_res =
    4757             :             std::vector<float>{-std::numeric_limits<float>::infinity(),
    4758             :                                -1.0f,
    4759             :                                1.0f,
    4760             :                                128.0f,
    4761             :                                32768.0f,
    4762           2 :                                std::numeric_limits<float>::infinity()};
    4763           1 :         EXPECT_EQ(res, expected_res);
    4764             : 
    4765           1 :         std::fill(res.begin(), res.end(), 0.0f);
    4766           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4767           1 :         EXPECT_EQ(res, expected_res);
    4768             :     }
    4769             : 
    4770             :     // Test complex<float>
    4771             :     {
    4772           2 :         std::vector<std::complex<float>> res;
    4773           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4774             :         const auto expected_res = std::vector<std::complex<float>>{
    4775             :             -std::numeric_limits<float>::infinity(),
    4776             :             -1.0f,
    4777             :             1.0f,
    4778             :             128.0f,
    4779             :             32768.0f,
    4780           2 :             std::numeric_limits<float>::infinity()};
    4781           1 :         EXPECT_EQ(res, expected_res);
    4782             : 
    4783           1 :         std::fill(res.begin(), res.end(), 0.0f);
    4784           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4785           1 :         EXPECT_EQ(res, expected_res);
    4786             :     }
    4787             : 
    4788             :     // Test double
    4789             :     {
    4790           2 :         std::vector<double> res;
    4791           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4792             :         const auto expected_res =
    4793           2 :             std::vector<double>{-1e300, -1, 1, 128, 32768, 1e300};
    4794           1 :         EXPECT_EQ(res, expected_res);
    4795             : 
    4796           1 :         std::fill(res.begin(), res.end(), 0);
    4797           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4798           1 :         EXPECT_EQ(res, expected_res);
    4799             :     }
    4800             : 
    4801             :     // Test complex<double>
    4802             :     {
    4803           2 :         std::vector<std::complex<double>> res;
    4804           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res), CE_None);
    4805             :         const auto expected_res =
    4806           2 :             std::vector<std::complex<double>>{-1e300, -1, 1, 128, 32768, 1e300};
    4807           1 :         EXPECT_EQ(res, expected_res);
    4808             : 
    4809           1 :         std::fill(res.begin(), res.end(), 0);
    4810           1 :         EXPECT_EQ(poDS->GetRasterBand(1)->ReadRaster(res.data()), CE_None);
    4811           1 :         EXPECT_EQ(res, expected_res);
    4812             :     }
    4813             : }
    4814             : 
    4815             : // Test GDALComputeRasterMinMaxLocation
    4816           4 : TEST_F(test_gdal, GDALComputeRasterMinMaxLocation)
    4817             : {
    4818           1 :     GDALDatasetH hDS = GDALOpen(GCORE_DATA_DIR "byte.tif", GA_ReadOnly);
    4819           1 :     ASSERT_NE(hDS, nullptr);
    4820           1 :     GDALRasterBandH hBand = GDALGetRasterBand(hDS, 1);
    4821             :     {
    4822           1 :         double dfMin = 0;
    4823           1 :         double dfMax = 0;
    4824           1 :         int nMinX = -1;
    4825           1 :         int nMinY = -1;
    4826           1 :         int nMaxX = -1;
    4827           1 :         int nMaxY = -1;
    4828           1 :         EXPECT_EQ(GDALComputeRasterMinMaxLocation(hBand, &dfMin, &dfMax, &nMinX,
    4829             :                                                   &nMinY, &nMaxX, &nMaxY),
    4830             :                   CE_None);
    4831           1 :         EXPECT_EQ(dfMin, 74.0);
    4832           1 :         EXPECT_EQ(dfMax, 255.0);
    4833           1 :         EXPECT_EQ(nMinX, 9);
    4834           1 :         EXPECT_EQ(nMinY, 17);
    4835           1 :         EXPECT_EQ(nMaxX, 2);
    4836           1 :         EXPECT_EQ(nMaxY, 18);
    4837           1 :         GByte val = 0;
    4838           1 :         EXPECT_EQ(GDALRasterIO(hBand, GF_Read, nMinX, nMinY, 1, 1, &val, 1, 1,
    4839             :                                GDT_Byte, 0, 0),
    4840             :                   CE_None);
    4841           1 :         EXPECT_EQ(val, 74);
    4842           1 :         EXPECT_EQ(GDALRasterIO(hBand, GF_Read, nMaxX, nMaxY, 1, 1, &val, 1, 1,
    4843             :                                GDT_Byte, 0, 0),
    4844             :                   CE_None);
    4845           1 :         EXPECT_EQ(val, 255);
    4846             :     }
    4847             :     {
    4848           1 :         int nMinX = -1;
    4849           1 :         int nMinY = -1;
    4850           1 :         EXPECT_EQ(GDALComputeRasterMinMaxLocation(hBand, nullptr, nullptr,
    4851             :                                                   &nMinX, &nMinY, nullptr,
    4852             :                                                   nullptr),
    4853             :                   CE_None);
    4854           1 :         EXPECT_EQ(nMinX, 9);
    4855           1 :         EXPECT_EQ(nMinY, 17);
    4856             :     }
    4857             :     {
    4858           1 :         int nMaxX = -1;
    4859           1 :         int nMaxY = -1;
    4860           1 :         EXPECT_EQ(GDALComputeRasterMinMaxLocation(hBand, nullptr, nullptr,
    4861             :                                                   nullptr, nullptr, &nMaxX,
    4862             :                                                   &nMaxY),
    4863             :                   CE_None);
    4864           1 :         EXPECT_EQ(nMaxX, 2);
    4865           1 :         EXPECT_EQ(nMaxY, 18);
    4866             :     }
    4867             :     {
    4868           1 :         EXPECT_EQ(GDALComputeRasterMinMaxLocation(hBand, nullptr, nullptr,
    4869             :                                                   nullptr, nullptr, nullptr,
    4870             :                                                   nullptr),
    4871             :                   CE_None);
    4872             :     }
    4873           1 :     GDALClose(hDS);
    4874             : }
    4875             : 
    4876             : // Test GDALComputeRasterMinMaxLocation
    4877           4 : TEST_F(test_gdal, GDALComputeRasterMinMaxLocation_byte_min_max_optim)
    4878             : {
    4879             :     GDALDatasetUniquePtr poDS(GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    4880           2 :                                   ->Create("", 1, 4, 1, GDT_Byte, nullptr));
    4881           1 :     std::array<uint8_t, 4> buffer = {
    4882             :         1,    //////////////////////////////////////////////////////////
    4883             :         0,    //////////////////////////////////////////////////////////
    4884             :         255,  //////////////////////////////////////////////////////////
    4885             :         1,    //////////////////////////////////////////////////////////
    4886             :     };
    4887             :     GDALRasterIOExtraArg sExtraArg;
    4888           1 :     INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    4889           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->RasterIO(
    4890             :                   GF_Write, 0, 0, 1, 4, buffer.data(), 1, 4, GDT_Byte,
    4891             :                   sizeof(uint8_t), 1 * sizeof(uint8_t), &sExtraArg),
    4892             :               CE_None);
    4893             : 
    4894           1 :     double dfMin = 0;
    4895           1 :     double dfMax = 0;
    4896           1 :     int nMinX = -1;
    4897           1 :     int nMinY = -1;
    4898           1 :     int nMaxX = -1;
    4899           1 :     int nMaxY = -1;
    4900           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->ComputeRasterMinMaxLocation(
    4901             :                   &dfMin, &dfMax, &nMinX, &nMinY, &nMaxX, &nMaxY),
    4902             :               CE_None);
    4903           1 :     EXPECT_EQ(dfMin, 0);
    4904           1 :     EXPECT_EQ(dfMax, 255);
    4905           1 :     EXPECT_EQ(nMinX, 0);
    4906           1 :     EXPECT_EQ(nMinY, 1);
    4907           1 :     EXPECT_EQ(nMaxX, 0);
    4908           1 :     EXPECT_EQ(nMaxY, 2);
    4909           1 : }
    4910             : 
    4911             : // Test GDALComputeRasterMinMaxLocation
    4912           4 : TEST_F(test_gdal, GDALComputeRasterMinMaxLocation_with_mask)
    4913             : {
    4914             :     GDALDatasetUniquePtr poDS(GDALDriver::FromHandle(GDALGetDriverByName("MEM"))
    4915           2 :                                   ->Create("", 2, 2, 1, GDT_Byte, nullptr));
    4916           1 :     std::array<uint8_t, 6> buffer = {
    4917             :         2, 10,  //////////////////////////////////////////////////////////
    4918             :         4, 20,  //////////////////////////////////////////////////////////
    4919             :     };
    4920             :     GDALRasterIOExtraArg sExtraArg;
    4921           1 :     INIT_RASTERIO_EXTRA_ARG(sExtraArg);
    4922           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->RasterIO(
    4923             :                   GF_Write, 0, 0, 2, 2, buffer.data(), 2, 2, GDT_Byte,
    4924             :                   sizeof(uint8_t), 2 * sizeof(uint8_t), &sExtraArg),
    4925             :               CE_None);
    4926             : 
    4927           1 :     poDS->GetRasterBand(1)->CreateMaskBand(0);
    4928           1 :     std::array<uint8_t, 6> buffer_mask = {
    4929             :         0, 255,  //////////////////////////////////////////////////////////
    4930             :         255, 0,  //////////////////////////////////////////////////////////
    4931             :     };
    4932           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->GetMaskBand()->RasterIO(
    4933             :                   GF_Write, 0, 0, 2, 2, buffer_mask.data(), 2, 2, GDT_Byte,
    4934             :                   sizeof(uint8_t), 2 * sizeof(uint8_t), &sExtraArg),
    4935             :               CE_None);
    4936             : 
    4937           1 :     double dfMin = 0;
    4938           1 :     double dfMax = 0;
    4939           1 :     int nMinX = -1;
    4940           1 :     int nMinY = -1;
    4941           1 :     int nMaxX = -1;
    4942           1 :     int nMaxY = -1;
    4943           1 :     EXPECT_EQ(poDS->GetRasterBand(1)->ComputeRasterMinMaxLocation(
    4944             :                   &dfMin, &dfMax, &nMinX, &nMinY, &nMaxX, &nMaxY),
    4945             :               CE_None);
    4946           1 :     EXPECT_EQ(dfMin, 4);
    4947           1 :     EXPECT_EQ(dfMax, 10);
    4948           1 :     EXPECT_EQ(nMinX, 0);
    4949           1 :     EXPECT_EQ(nMinY, 1);
    4950           1 :     EXPECT_EQ(nMaxX, 1);
    4951           1 :     EXPECT_EQ(nMaxY, 0);
    4952           1 : }
    4953             : 
    4954           4 : TEST_F(test_gdal, GDALTranspose2D)
    4955             : {
    4956           1 :     constexpr int COUNT = 6;
    4957           1 :     const GByte abyData[] = {1, 2, 3, 4, 5, 6};
    4958             :     GByte abySrcData[COUNT * 2 * sizeof(double)];
    4959             :     GByte abyDstData[COUNT * 2 * sizeof(double)];
    4960             :     GByte abyDstAsByteData[COUNT * 2 * sizeof(double)];
    4961          15 :     for (int eSrcDTInt = GDT_Byte; eSrcDTInt < GDT_TypeCount; ++eSrcDTInt)
    4962             :     {
    4963          14 :         const auto eSrcDT = static_cast<GDALDataType>(eSrcDTInt);
    4964          14 :         GDALCopyWords(abyData, GDT_Byte, 1, abySrcData, eSrcDT,
    4965             :                       GDALGetDataTypeSizeBytes(eSrcDT), COUNT);
    4966         210 :         for (int eDstDTInt = GDT_Byte; eDstDTInt < GDT_TypeCount; ++eDstDTInt)
    4967             :         {
    4968         196 :             const auto eDstDT = static_cast<GDALDataType>(eDstDTInt);
    4969         196 :             memset(abyDstData, 0, sizeof(abyDstData));
    4970         196 :             GDALTranspose2D(abySrcData, eSrcDT, abyDstData, eDstDT, 3, 2);
    4971             : 
    4972         196 :             memset(abyDstAsByteData, 0, sizeof(abyDstAsByteData));
    4973         196 :             GDALCopyWords(abyDstData, eDstDT, GDALGetDataTypeSizeBytes(eDstDT),
    4974             :                           abyDstAsByteData, GDT_Byte, 1, COUNT);
    4975             : 
    4976         196 :             EXPECT_EQ(abyDstAsByteData[0], 1)
    4977           0 :                 << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT;
    4978         196 :             EXPECT_EQ(abyDstAsByteData[1], 4)
    4979           0 :                 << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT;
    4980         196 :             EXPECT_EQ(abyDstAsByteData[2], 2)
    4981           0 :                 << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT;
    4982         196 :             EXPECT_EQ(abyDstAsByteData[3], 5)
    4983           0 :                 << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT;
    4984         196 :             EXPECT_EQ(abyDstAsByteData[4], 3)
    4985           0 :                 << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT;
    4986         196 :             EXPECT_EQ(abyDstAsByteData[5], 6)
    4987           0 :                 << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT;
    4988             :         }
    4989             :     }
    4990           1 : }
    4991             : 
    4992           4 : TEST_F(test_gdal, GDALTranspose2D_Byte_optims)
    4993             : {
    4994           2 :     std::vector<GByte> in;
    4995         324 :     for (int i = 0; i < 19 * 17; ++i)
    4996         323 :         in.push_back(static_cast<GByte>(i % 256));
    4997             : 
    4998           2 :     std::vector<GByte> out(in.size());
    4999             : 
    5000             :     // SSSE3 optim (16x16) blocks
    5001             :     {
    5002           1 :         constexpr int W = 19;
    5003           1 :         constexpr int H = 17;
    5004           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5005          18 :         for (int y = 0; y < H; ++y)
    5006             :         {
    5007         340 :             for (int x = 0; x < W; ++x)
    5008             :             {
    5009         323 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5010             :             }
    5011             :         }
    5012             :     }
    5013             : 
    5014             :     // Optim H = 2 with W < 16
    5015             :     {
    5016           1 :         constexpr int W = 15;
    5017           1 :         constexpr int H = 2;
    5018           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5019           3 :         for (int y = 0; y < H; ++y)
    5020             :         {
    5021          32 :             for (int x = 0; x < W; ++x)
    5022             :             {
    5023          30 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5024             :             }
    5025             :         }
    5026             :     }
    5027             : 
    5028             :     // Optim H = 2 with W >= 16
    5029             :     {
    5030           1 :         constexpr int W = 19;
    5031           1 :         constexpr int H = 2;
    5032           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5033           3 :         for (int y = 0; y < H; ++y)
    5034             :         {
    5035          40 :             for (int x = 0; x < W; ++x)
    5036             :             {
    5037          38 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5038             :             }
    5039             :         }
    5040             :     }
    5041             : 
    5042             :     // SSSE3 optim H = 3 with W < 16
    5043             :     {
    5044           1 :         constexpr int W = 15;
    5045           1 :         constexpr int H = 3;
    5046           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5047           4 :         for (int y = 0; y < H; ++y)
    5048             :         {
    5049          48 :             for (int x = 0; x < W; ++x)
    5050             :             {
    5051          45 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5052             :             }
    5053             :         }
    5054             :     }
    5055             : 
    5056             :     // SSSE3 optim H = 3 with W >= 16
    5057             :     {
    5058           1 :         constexpr int W = 19;
    5059           1 :         constexpr int H = 3;
    5060           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5061           4 :         for (int y = 0; y < H; ++y)
    5062             :         {
    5063          60 :             for (int x = 0; x < W; ++x)
    5064             :             {
    5065          57 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5066             :             }
    5067             :         }
    5068             :     }
    5069             : 
    5070             :     // Optim H = 4 with H < 16
    5071             :     {
    5072           1 :         constexpr int W = 15;
    5073           1 :         constexpr int H = 4;
    5074           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5075           5 :         for (int y = 0; y < H; ++y)
    5076             :         {
    5077          64 :             for (int x = 0; x < W; ++x)
    5078             :             {
    5079          60 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5080             :             }
    5081             :         }
    5082             :     }
    5083             : 
    5084             :     // Optim H = 4 with H >= 16
    5085             :     {
    5086           1 :         constexpr int W = 19;
    5087           1 :         constexpr int H = 4;
    5088           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5089           5 :         for (int y = 0; y < H; ++y)
    5090             :         {
    5091          80 :             for (int x = 0; x < W; ++x)
    5092             :             {
    5093          76 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5094             :             }
    5095             :         }
    5096             :     }
    5097             : 
    5098             :     // SSSE3 optim H = 5 with W < 16
    5099             :     {
    5100           1 :         constexpr int W = 15;
    5101           1 :         constexpr int H = 5;
    5102           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5103           6 :         for (int y = 0; y < H; ++y)
    5104             :         {
    5105          80 :             for (int x = 0; x < W; ++x)
    5106             :             {
    5107          75 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5108             :             }
    5109             :         }
    5110             :     }
    5111             : 
    5112             :     // SSSE3 optim H = 5 with W >= 16
    5113             :     {
    5114           1 :         constexpr int W = 19;
    5115           1 :         constexpr int H = 5;
    5116           1 :         GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H);
    5117           6 :         for (int y = 0; y < H; ++y)
    5118             :         {
    5119         100 :             for (int x = 0; x < W; ++x)
    5120             :             {
    5121          95 :                 EXPECT_EQ(out[x * H + y], in[y * W + x]);
    5122             :             }
    5123             :         }
    5124             :     }
    5125           1 : }
    5126             : 
    5127           4 : TEST_F(test_gdal, GDALExpandPackedBitsToByteAt0Or1)
    5128             : {
    5129           1 :     unsigned next = 1;
    5130         141 :     const auto badRand = [&next]()
    5131             :     {
    5132         141 :         next = static_cast<unsigned>(static_cast<uint64_t>(next) * 1103515245 +
    5133             :                                      12345);
    5134         141 :         return next;
    5135           1 :     };
    5136             : 
    5137           1 :     constexpr int BITS_PER_BYTE = 8;
    5138           1 :     constexpr int SSE_REGISTER_SIZE_IN_BYTES = 16;
    5139           1 :     constexpr int LESS_THAN_8BITS = 5;
    5140             :     std::vector<GByte> expectedOut(SSE_REGISTER_SIZE_IN_BYTES * BITS_PER_BYTE +
    5141           2 :                                    BITS_PER_BYTE + LESS_THAN_8BITS);
    5142           1 :     std::vector<GByte> in((expectedOut.size() + BITS_PER_BYTE - 1) /
    5143           3 :                           BITS_PER_BYTE);
    5144         142 :     for (int i = 0; i < static_cast<int>(expectedOut.size()); ++i)
    5145             :     {
    5146         141 :         expectedOut[i] = (badRand() % 2) == 0 ? 0 : 1;
    5147         141 :         if (expectedOut[i])
    5148             :         {
    5149          70 :             in[i / BITS_PER_BYTE] = static_cast<GByte>(
    5150          70 :                 in[i / BITS_PER_BYTE] |
    5151          70 :                 (1 << (BITS_PER_BYTE - 1 - (i % BITS_PER_BYTE))));
    5152             :         }
    5153             :     }
    5154             : 
    5155           2 :     std::vector<GByte> out(expectedOut.size());
    5156           1 :     GDALExpandPackedBitsToByteAt0Or1(in.data(), out.data(), out.size());
    5157             : 
    5158           1 :     EXPECT_EQ(out, expectedOut);
    5159           1 : }
    5160             : 
    5161           4 : TEST_F(test_gdal, GDALExpandPackedBitsToByteAt0Or255)
    5162             : {
    5163           1 :     unsigned next = 1;
    5164         141 :     const auto badRand = [&next]()
    5165             :     {
    5166         141 :         next = static_cast<unsigned>(static_cast<uint64_t>(next) * 1103515245 +
    5167             :                                      12345);
    5168         141 :         return next;
    5169           1 :     };
    5170             : 
    5171           1 :     constexpr int BITS_PER_BYTE = 8;
    5172           1 :     constexpr int SSE_REGISTER_SIZE_IN_BYTES = 16;
    5173           1 :     constexpr int LESS_THAN_8BITS = 5;
    5174             :     std::vector<GByte> expectedOut(SSE_REGISTER_SIZE_IN_BYTES * BITS_PER_BYTE +
    5175           2 :                                    BITS_PER_BYTE + LESS_THAN_8BITS);
    5176           1 :     std::vector<GByte> in((expectedOut.size() + BITS_PER_BYTE - 1) /
    5177           3 :                           BITS_PER_BYTE);
    5178         142 :     for (int i = 0; i < static_cast<int>(expectedOut.size()); ++i)
    5179             :     {
    5180         141 :         expectedOut[i] = (badRand() % 2) == 0 ? 0 : 255;
    5181         141 :         if (expectedOut[i])
    5182             :         {
    5183          70 :             in[i / BITS_PER_BYTE] = static_cast<GByte>(
    5184          70 :                 in[i / BITS_PER_BYTE] |
    5185          70 :                 (1 << (BITS_PER_BYTE - 1 - (i % BITS_PER_BYTE))));
    5186             :         }
    5187             :     }
    5188             : 
    5189           2 :     std::vector<GByte> out(expectedOut.size());
    5190           1 :     GDALExpandPackedBitsToByteAt0Or255(in.data(), out.data(), out.size());
    5191             : 
    5192           1 :     EXPECT_EQ(out, expectedOut);
    5193           1 : }
    5194             : 
    5195             : }  // namespace

Generated by: LCOV version 1.14