diff --git a/.github/workflows/asan/test.sh b/.github/workflows/asan/test.sh index b51a96ef8fcc..acf8a6576fd7 100755 --- a/.github/workflows/asan/test.sh +++ b/.github/workflows/asan/test.sh @@ -4,8 +4,8 @@ set -ex . ../scripts/setdevenv.sh -export LD_LIBRARY_PATH=/usr/lib/llvm-14/lib/clang/14.0.0/lib/linux:${LD_LIBRARY_PATH} -export PATH=/usr/lib/llvm-14/bin:${PATH} +export LD_LIBRARY_PATH=/usr/lib/llvm-18/lib/clang/18.0.0/lib/linux:${LD_LIBRARY_PATH} +export PATH=/usr/lib/llvm-18/bin:${PATH} export SKIP_MEM_INTENSIVE_TEST=YES export SKIP_VIRTUALMEM=YES export LD_PRELOAD=$(clang -print-file-name=libclang_rt.asan-x86_64.so) diff --git a/.github/workflows/linux_build.yml b/.github/workflows/linux_build.yml index 6fd0cdc474d2..6b5aa0de2c13 100644 --- a/.github/workflows/linux_build.yml +++ b/.github/workflows/linux_build.yml @@ -100,13 +100,13 @@ jobs: test_script: test.sh os: ubuntu-22.04 - - name: Ubuntu 22.04, clang ASAN + - name: Ubuntu 24.04, clang ASAN id: asan travis_branch: sanitize - container: ubuntu_22.04 + container: ubuntu_24.04 build_script: build.sh test_script: test.sh - os: ubuntu-22.04 + os: ubuntu-24.04 - name: Ubuntu 20.04, gcc id: ubuntu_20.04 diff --git a/.github/workflows/ubuntu_24.04/expected_gdalinfo_formats.txt b/.github/workflows/ubuntu_24.04/expected_gdalinfo_formats.txt index 396d0730f97e..db8d5841a8e3 100644 --- a/.github/workflows/ubuntu_24.04/expected_gdalinfo_formats.txt +++ b/.github/workflows/ubuntu_24.04/expected_gdalinfo_formats.txt @@ -5,6 +5,7 @@ Supported Formats: (ro:read-only, rw:read-write, +:update, v:virtual-I/O s:subda SNAP_TIFF -raster- (rov): Sentinel Application Processing GeoTIFF GTiff -raster- (rw+vs): GeoTIFF (*.tif, *.tiff) COG -raster- (wv): Cloud optimized GeoTIFF generator (*.tif, *.tiff) + LIBERTIFF -raster- (rov): GeoTIFF (using LIBERTIFF library) (*.tif, *.tiff) NITF -raster- (rw+vs): National Imagery Transmission Format (*.ntf) RPFTOC -raster- (rovs): Raster Product Format TOC format (*.toc) ECRGTOC -raster- (rovs): ECRG TOC format (*.xml) diff --git a/.github/workflows/windows_conda_expected_gdalinfo_formats.txt b/.github/workflows/windows_conda_expected_gdalinfo_formats.txt index f37c96eec515..5b0cb5089e3d 100644 --- a/.github/workflows/windows_conda_expected_gdalinfo_formats.txt +++ b/.github/workflows/windows_conda_expected_gdalinfo_formats.txt @@ -5,6 +5,7 @@ Supported Formats: (ro:read-only, rw:read-write, +:update, v:virtual-I/O s:subda SNAP_TIFF -raster- (rov): Sentinel Application Processing GeoTIFF GTiff -raster- (rw+vs): GeoTIFF (*.tif, *.tiff) COG -raster- (wv): Cloud optimized GeoTIFF generator (*.tif, *.tiff) + LIBERTIFF -raster- (rov): GeoTIFF (using LIBERTIFF library) (*.tif, *.tiff) NITF -raster- (rw+vs): National Imagery Transmission Format (*.ntf) RPFTOC -raster- (rovs): Raster Product Format TOC format (*.toc) ECRGTOC -raster- (rovs): ECRG TOC format (*.xml) diff --git a/alg/gdal_interpolateatpoint.h b/alg/gdal_interpolateatpoint.h index e92deaf7683e..735265dea43f 100644 --- a/alg/gdal_interpolateatpoint.h +++ b/alg/gdal_interpolateatpoint.h @@ -28,17 +28,18 @@ using DoublePointsCache = lru11::Cache>>; -class GDALDoublePointsCache +class CPL_DLL GDALDoublePointsCache { public: std::unique_ptr cache{}; }; -bool GDALInterpolateAtPoint(GDALRasterBand *pBand, - GDALRIOResampleAlg eResampleAlg, - std::unique_ptr &cache, - const double dfXIn, const double dfYIn, - double *pdfOutputReal, double *pdfOutputImag); +bool CPL_DLL GDALInterpolateAtPoint(GDALRasterBand *pBand, + GDALRIOResampleAlg eResampleAlg, + std::unique_ptr &cache, + const double dfXIn, const double dfYIn, + double *pdfOutputReal, + double *pdfOutputImag); /*! @endcond */ diff --git a/apps/CMakeLists.txt b/apps/CMakeLists.txt index 6fb96bc274e2..e37a04fc79f1 100644 --- a/apps/CMakeLists.txt +++ b/apps/CMakeLists.txt @@ -11,6 +11,7 @@ add_library( gdalalg_main.cpp gdalalg_pipeline.cpp gdalalg_raster.cpp + gdalalg_raster_buildvrt.cpp gdalalg_raster_info.cpp gdalalg_raster_convert.cpp gdalalg_raster_edit.cpp diff --git a/apps/gdal.cpp b/apps/gdal.cpp index 88ea2e4dadce..9811118f18c0 100644 --- a/apps/gdal.cpp +++ b/apps/gdal.cpp @@ -17,12 +17,96 @@ #include +// #define DEBUG_COMPLETION + +/************************************************************************/ +/* EmitCompletion() */ +/************************************************************************/ + +/** Return on stdout a space-separated list of choices for bash completion */ +static void EmitCompletion(std::unique_ptr rootAlg, + const std::vector &argsIn) +{ +#ifdef DEBUG_COMPLETION + for (size_t i = 0; i < argsIn.size(); ++i) + fprintf(stderr, "arg[%d]='%s'\n", static_cast(i), + argsIn[i].c_str()); +#endif + + std::vector args = argsIn; + + std::string ret; + const auto addSpace = [&ret]() + { + if (!ret.empty()) + ret += " "; + }; + + if (!args.empty() && + (args.back() == "--config" || + STARTS_WITH(args.back().c_str(), "--config=") || + (args.size() >= 2 && args[args.size() - 2] == "--config"))) + { + if (args.back() == "--config=" || args.back().back() != '=') + { + CPLStringList aosConfigOptions(CPLGetKnownConfigOptions()); + for (const char *pszOpt : cpl::Iterate(aosConfigOptions)) + { + addSpace(); + ret += pszOpt; + ret += '='; + } + printf("%s", ret.c_str()); + } + return; + } + + // Get inner-most algorithm + bool showAllOptions = true; + auto curAlg = std::move(rootAlg); + while (!args.empty() && !args.front().empty() && args.front()[0] != '-') + { + auto subAlg = curAlg->InstantiateSubAlgorithm(args.front()); + if (!subAlg) + break; + showAllOptions = false; + args.erase(args.begin()); + curAlg = std::move(subAlg); + } + + for (const auto &choice : curAlg->GetAutoComplete(args, showAllOptions)) + { + addSpace(); + ret += CPLString(choice).replaceAll(" ", "\\ "); + } + +#ifdef DEBUG_COMPLETION + fprintf(stderr, "ret = '%s'\n", ret.c_str()); +#endif + if (!ret.empty()) + printf("%s", ret.c_str()); +} + /************************************************************************/ /* main() */ /************************************************************************/ MAIN_START(argc, argv) { + auto alg = GDALGlobalAlgorithmRegistry::GetSingleton().Instantiate( + GDALGlobalAlgorithmRegistry::ROOT_ALG_NAME); + assert(alg); + + if (argc >= 3 && strcmp(argv[1], "completion") == 0) + { + GDALAllRegister(); + + // Process lines like "gdal completion gdal raster" + EmitCompletion(std::move(alg), + std::vector(argv + 3, argv + argc)); + return 0; + } + EarlySetConfigOptions(argc, argv); /* -------------------------------------------------------------------- */ @@ -31,14 +115,12 @@ MAIN_START(argc, argv) /* -------------------------------------------------------------------- */ GDALAllRegister(); + argc = GDALGeneralCmdLineProcessor( argc, &argv, GDAL_OF_RASTER | GDAL_OF_VECTOR | GDAL_OF_MULTIDIM_RASTER); if (argc < 1) return (-argc); - auto alg = GDALGlobalAlgorithmRegistry::GetSingleton().Instantiate( - GDALGlobalAlgorithmRegistry::ROOT_ALG_NAME); - assert(alg); alg->SetCallPath(std::vector{argv[0]}); std::vector args; for (int i = 1; i < argc; ++i) diff --git a/apps/gdalalg_abstract_pipeline.h b/apps/gdalalg_abstract_pipeline.h new file mode 100644 index 000000000000..5ac5a2ea5c85 --- /dev/null +++ b/apps/gdalalg_abstract_pipeline.h @@ -0,0 +1,238 @@ +/****************************************************************************** + * + * Project: GDAL + * Purpose: gdal "raster/vector pipeline" subcommand + * Author: Even Rouault + * + ****************************************************************************** + * Copyright (c) 2024, Even Rouault + * + * SPDX-License-Identifier: MIT + ****************************************************************************/ + +#ifndef GDALALG_ABSTRACT_PIPELINE_INCLUDED +#define GDALALG_ABSTRACT_PIPELINE_INCLUDED + +//! @cond Doxygen_Suppress + +#include "cpl_conv.h" +#include "cpl_json.h" +#include "gdalalgorithm.h" + +template +class GDALAbstractPipelineAlgorithm CPL_NON_FINAL : public StepAlgorithm +{ + public: + std::vector GetAutoComplete(std::vector &args, + bool /* showAllOptions*/) override; + + bool Finalize() override; + + std::string GetUsageAsJSON() const override; + + /* cppcheck-suppress functionStatic */ + void SetDataset(GDALDataset *) + { + } + + protected: + GDALAbstractPipelineAlgorithm(const std::string &name, + const std::string &description, + const std::string &helpURL, + bool standaloneStep) + : StepAlgorithm(name, description, helpURL, standaloneStep) + { + } + + virtual GDALArgDatasetValue &GetOutputDataset() = 0; + + std::string m_pipeline{}; + + std::unique_ptr GetStepAlg(const std::string &name) const; + + GDALAlgorithmRegistry m_stepRegistry{}; + std::vector> m_steps{}; + + private: + bool RunStep(GDALProgressFunc pfnProgress, void *pProgressData) override; +}; + +/************************************************************************/ +/* GDALAbstractPipelineAlgorithm::GetStepAlg() */ +/************************************************************************/ + +template +std::unique_ptr +GDALAbstractPipelineAlgorithm::GetStepAlg( + const std::string &name) const +{ + auto alg = m_stepRegistry.Instantiate(name); + return std::unique_ptr( + cpl::down_cast(alg.release())); +} + +/************************************************************************/ +/* GDALAbstractPipelineAlgorithm::GetAutoComplete() */ +/************************************************************************/ + +template +std::vector +GDALAbstractPipelineAlgorithm::GetAutoComplete( + std::vector &args, bool /* showAllOptions*/) +{ + std::vector ret; + if (args.size() <= 1) + { + if (args.empty() || args.front() != "read") + ret.push_back("read"); + } + else if (args.back() == "!" || args[args.size() - 2] == "!") + { + for (const std::string &name : m_stepRegistry.GetNames()) + { + if (name != "read") + { + ret.push_back(name); + } + } + } + else + { + std::string lastStep = "read"; + std::vector lastArgs; + for (size_t i = 1; i < args.size(); ++i) + { + lastArgs.push_back(args[i]); + if (i + 1 < args.size() && args[i] == "!") + { + ++i; + lastArgs.clear(); + lastStep = args[i]; + } + } + + auto curAlg = GetStepAlg(lastStep); + if (curAlg) + { + ret = + curAlg->GetAutoComplete(lastArgs, /* showAllOptions = */ false); + } + } + return ret; +} + +/************************************************************************/ +/* GDALAbstractPipelineAlgorithm::RunStep() */ +/************************************************************************/ + +template +bool GDALAbstractPipelineAlgorithm::RunStep( + GDALProgressFunc pfnProgress, void *pProgressData) +{ + if (m_steps.empty()) + { + // If invoked programmatically, not from the command line. + + if (m_pipeline.empty()) + { + StepAlgorithm::ReportError(CE_Failure, CPLE_AppDefined, + "'pipeline' argument not set"); + return false; + } + + const CPLStringList aosTokens(CSLTokenizeString(m_pipeline.c_str())); + if (!this->ParseCommandLineArguments(aosTokens)) + return false; + } + + GDALDataset *poCurDS = nullptr; + for (size_t i = 0; i < m_steps.size(); ++i) + { + auto &step = m_steps[i]; + if (i > 0) + { + if (step->m_inputDataset.GetDatasetRef()) + { + // Shouldn't happen + StepAlgorithm::ReportError( + CE_Failure, CPLE_AppDefined, + "Step nr %d (%s) has already an input dataset", + static_cast(i), step->GetName().c_str()); + return false; + } + step->m_inputDataset.Set(poCurDS); + } + if (i + 1 < m_steps.size() && step->m_outputDataset.GetDatasetRef()) + { + // Shouldn't happen + StepAlgorithm::ReportError( + CE_Failure, CPLE_AppDefined, + "Step nr %d (%s) has already an output dataset", + static_cast(i), step->GetName().c_str()); + return false; + } + if (!step->Run(i < m_steps.size() - 1 ? nullptr : pfnProgress, + i < m_steps.size() - 1 ? nullptr : pProgressData)) + { + return false; + } + poCurDS = step->m_outputDataset.GetDatasetRef(); + if (!poCurDS) + { + StepAlgorithm::ReportError( + CE_Failure, CPLE_AppDefined, + "Step nr %d (%s) failed to produce an output dataset", + static_cast(i), step->GetName().c_str()); + return false; + } + } + + if (!GetOutputDataset().GetDatasetRef()) + { + GetOutputDataset().Set(poCurDS); + } + + return true; +} + +/************************************************************************/ +/* GDALAbstractPipelineAlgorithm::Finalize() */ +/************************************************************************/ + +template +bool GDALAbstractPipelineAlgorithm::Finalize() +{ + bool ret = GDALAlgorithm::Finalize(); + for (auto &step : m_steps) + { + ret = step->Finalize() && ret; + } + return ret; +} + +/************************************************************************/ +/* GDALAbstractPipelineAlgorithm::GetUsageAsJSON() */ +/************************************************************************/ + +template +std::string GDALAbstractPipelineAlgorithm::GetUsageAsJSON() const +{ + CPLJSONDocument oDoc; + oDoc.LoadMemory(GDALAlgorithm::GetUsageAsJSON()); + + CPLJSONArray jPipelineSteps; + for (const std::string &name : m_stepRegistry.GetNames()) + { + auto alg = GetStepAlg(name); + CPLJSONDocument oStepDoc; + oStepDoc.LoadMemory(alg->GetUsageAsJSON()); + jPipelineSteps.Add(oStepDoc.GetRoot()); + } + oDoc.GetRoot().Add("pipeline_algorithms", jPipelineSteps); + + return oDoc.SaveAsString(); +} + +//! @endcond + +#endif diff --git a/apps/gdalalg_raster.cpp b/apps/gdalalg_raster.cpp index 22fed2833aa6..b2975197a303 100644 --- a/apps/gdalalg_raster.cpp +++ b/apps/gdalalg_raster.cpp @@ -12,6 +12,7 @@ #include "gdalalgorithm.h" +#include "gdalalg_raster_buildvrt.h" #include "gdalalg_raster_info.h" #include "gdalalg_raster_convert.h" #include "gdalalg_raster_edit.h" @@ -41,6 +42,7 @@ class GDALRasterAlgorithm final : public GDALAlgorithm RegisterSubAlgorithm(); RegisterSubAlgorithm(); RegisterSubAlgorithm(); + RegisterSubAlgorithm(); } private: diff --git a/apps/gdalalg_raster_buildvrt.cpp b/apps/gdalalg_raster_buildvrt.cpp new file mode 100644 index 000000000000..e8d32df67e5d --- /dev/null +++ b/apps/gdalalg_raster_buildvrt.cpp @@ -0,0 +1,277 @@ +/****************************************************************************** + * + * Project: GDAL + * Purpose: gdal "raster buildvrt" subcommand + * Author: Even Rouault + * + ****************************************************************************** + * Copyright (c) 2024, Even Rouault + * + * SPDX-License-Identifier: MIT + ****************************************************************************/ + +#include "gdalalg_raster_buildvrt.h" + +#include "cpl_conv.h" +#include "cpl_vsi_virtual.h" + +#include "gdal_priv.h" +#include "gdal_utils.h" + +//! @cond Doxygen_Suppress + +#ifndef _ +#define _(x) (x) +#endif + +/************************************************************************/ +/* GDALRasterBuildVRTAlgorithm::GDALRasterBuildVRTAlgorithm() */ +/************************************************************************/ + +GDALRasterBuildVRTAlgorithm::GDALRasterBuildVRTAlgorithm() + : GDALAlgorithm(NAME, DESCRIPTION, HELP_URL) +{ + AddProgressArg(); + AddArg(GDAL_ARG_NAME_INPUT, 'i', + _("Input raster datasets (or specify a @ to point to a " + "file containing filenames)"), + &m_inputDatasets, GDAL_OF_RASTER) + .SetPositional() + .SetMinCount(1) + .SetAutoOpenDataset(false) + .SetMetaVar("INPUTS"); + AddOutputDatasetArg(&m_outputDataset, GDAL_OF_RASTER); + AddCreationOptionsArg(&m_creationOptions); + AddArg("band", 'b', _("Specify input band(s) number."), &m_bands); + AddArg("separate", 0, _("Place each input file into a separate band."), + &m_separate); + AddOverwriteArg(&m_overwrite); + { + auto &arg = AddArg("resolution", 0, + _("Target resolution (in destination CRS units)"), + &m_resolution) + .SetMetaVar(",|average|highest|lowest"); + arg.AddValidationAction( + [this, &arg]() + { + const std::string val = arg.Get(); + if (val != "average" && val != "highest" && val != "lowest") + { + const auto aosTokens = + CPLStringList(CSLTokenizeString2(val.c_str(), ",", 0)); + if (aosTokens.size() != 2 || + CPLGetValueType(aosTokens[0]) == CPL_VALUE_STRING || + CPLGetValueType(aosTokens[1]) == CPL_VALUE_STRING || + CPLAtof(aosTokens[0]) <= 0 || + CPLAtof(aosTokens[1]) <= 0) + { + ReportError(CE_Failure, CPLE_AppDefined, + "resolution: two comma separated positive " + "values should be provided, or 'average', " + "'highest' or 'lowest'"); + return false; + } + } + return true; + }); + } + AddBBOXArg(&m_bbox, _("Target bounding box as xmin,ymin,xmax,ymax (in " + "destination CRS units)")); + AddArg("target-aligned-pixels", 0, + _("Round target extent to target resolution"), + &m_targetAlignedPixels) + .AddHiddenAlias("tap"); + AddArg("srcnodata", 0, _("Set nodata values for input bands."), + &m_srcNoData) + .SetMinCount(1) + .SetRepeatedArgAllowed(false); + AddArg("vrtnodata", 0, _("Set nodata values at the VRT band level."), + &m_vrtNoData) + .SetMinCount(1) + .SetRepeatedArgAllowed(false); + AddArg("hidenodata", 0, _("Makes the VRT band not report the NoData."), + &m_hideNoData); + AddArg("addalpha", 0, + _("Adds an alpha mask band to the VRT when the source raster have " + "none."), + &m_addAlpha); +} + +/************************************************************************/ +/* GDALRasterBuildVRTAlgorithm::RunImpl() */ +/************************************************************************/ + +bool GDALRasterBuildVRTAlgorithm::RunImpl(GDALProgressFunc pfnProgress, + void *pProgressData) +{ + if (m_outputDataset.GetDatasetRef()) + { + ReportError(CE_Failure, CPLE_NotSupported, + "gdal raster buildvrt does not support outputting to an " + "already opened output dataset"); + return false; + } + + std::vector ahInputDatasets; + CPLStringList aosInputDatasetNames; + bool foundByRef = false; + bool foundByName = false; + for (auto &ds : m_inputDatasets) + { + if (ds.GetDatasetRef()) + { + foundByRef = true; + ahInputDatasets.push_back( + GDALDataset::ToHandle(ds.GetDatasetRef())); + } + else if (!ds.GetName().empty()) + { + foundByName = true; + if (ds.GetName()[0] == '@') + { + auto f = VSIVirtualHandleUniquePtr( + VSIFOpenL(ds.GetName().c_str() + 1, "r")); + if (!f) + { + ReportError(CE_Failure, CPLE_FileIO, "Cannot open %s", + ds.GetName().c_str() + 1); + return false; + } + while (const char *filename = CPLReadLineL(f.get())) + { + aosInputDatasetNames.push_back(filename); + } + } + else if (ds.GetName().find_first_of("*?[") != std::string::npos) + { + CPLStringList aosMatches(VSIGlob(ds.GetName().c_str(), nullptr, + pfnProgress, pProgressData)); + for (const char *pszStr : aosMatches) + { + aosInputDatasetNames.push_back(pszStr); + } + } + else + { + aosInputDatasetNames.push_back(ds.GetName().c_str()); + } + } + } + if (foundByName && foundByRef) + { + ReportError(CE_Failure, CPLE_NotSupported, + "Input datasets should be provided either all by reference " + "or all by name"); + return false; + } + + VSIStatBufL sStat; + if (!m_overwrite && !m_outputDataset.GetName().empty() && + (VSIStatL(m_outputDataset.GetName().c_str(), &sStat) == 0 || + std::unique_ptr( + GDALDataset::Open(m_outputDataset.GetName().c_str())))) + { + ReportError(CE_Failure, CPLE_AppDefined, + "File '%s' already exists. Specify the --overwrite " + "option to overwrite it.", + m_outputDataset.GetName().c_str()); + return false; + } + + CPLStringList aosOptions; + if (!m_resolution.empty()) + { + const auto aosTokens = + CPLStringList(CSLTokenizeString2(m_resolution.c_str(), ",", 0)); + if (aosTokens.size() == 2) + { + aosOptions.push_back("-tr"); + aosOptions.push_back(aosTokens[0]); + aosOptions.push_back(aosTokens[1]); + } + else + { + aosOptions.push_back("-resolution"); + aosOptions.push_back(m_resolution); + } + } + if (!m_bbox.empty()) + { + aosOptions.push_back("-te"); + aosOptions.push_back(CPLSPrintf("%.17g", m_bbox[0])); + aosOptions.push_back(CPLSPrintf("%.17g", m_bbox[1])); + aosOptions.push_back(CPLSPrintf("%.17g", m_bbox[2])); + aosOptions.push_back(CPLSPrintf("%.17g", m_bbox[3])); + } + if (m_targetAlignedPixels) + { + aosOptions.push_back("-tap"); + } + if (!m_srcNoData.empty()) + { + aosOptions.push_back("-srcnodata"); + std::string s; + for (double v : m_srcNoData) + { + if (!s.empty()) + s += " "; + s += CPLSPrintf("%.17g", v); + } + aosOptions.push_back(s); + } + if (!m_vrtNoData.empty()) + { + aosOptions.push_back("-vrtnodata"); + std::string s; + for (double v : m_vrtNoData) + { + if (!s.empty()) + s += " "; + s += CPLSPrintf("%.17g", v); + } + aosOptions.push_back(s); + } + if (m_separate) + { + aosOptions.push_back("-separate"); + } + for (const auto &co : m_creationOptions) + { + aosOptions.push_back("-co"); + aosOptions.push_back(co); + } + for (const int b : m_bands) + { + aosOptions.push_back("-b"); + aosOptions.push_back(CPLSPrintf("%d", b)); + } + if (m_addAlpha) + { + aosOptions.push_back("-addalpha"); + } + if (m_hideNoData) + { + aosOptions.push_back("-hidenodata"); + } + + GDALBuildVRTOptions *psOptions = + GDALBuildVRTOptionsNew(aosOptions.List(), nullptr); + GDALBuildVRTOptionsSetProgress(psOptions, pfnProgress, pProgressData); + + auto poOutDS = std::unique_ptr(GDALDataset::FromHandle( + GDALBuildVRT(m_outputDataset.GetName().c_str(), + foundByName ? aosInputDatasetNames.size() + : static_cast(m_inputDatasets.size()), + ahInputDatasets.empty() ? nullptr : ahInputDatasets.data(), + aosInputDatasetNames.List(), psOptions, nullptr))); + GDALBuildVRTOptionsFree(psOptions); + const bool bOK = poOutDS != nullptr; + if (bOK) + { + m_outputDataset.Set(std::move(poOutDS)); + } + + return bOK; +} + +//! @endcond diff --git a/apps/gdalalg_raster_buildvrt.h b/apps/gdalalg_raster_buildvrt.h new file mode 100644 index 000000000000..d01a562431eb --- /dev/null +++ b/apps/gdalalg_raster_buildvrt.h @@ -0,0 +1,59 @@ +/****************************************************************************** + * + * Project: GDAL + * Purpose: gdal "raster buildvrt" subcommand + * Author: Even Rouault + * + ****************************************************************************** + * Copyright (c) 2024, Even Rouault + * + * SPDX-License-Identifier: MIT + ****************************************************************************/ + +#ifndef GDALALG_RASTER_BUILDVRT_INCLUDED +#define GDALALG_RASTER_BUILDVRT_INCLUDED + +#include "gdalalgorithm.h" + +//! @cond Doxygen_Suppress + +/************************************************************************/ +/* GDALRasterBuildVRTAlgorithm */ +/************************************************************************/ + +class GDALRasterBuildVRTAlgorithm final : public GDALAlgorithm +{ + public: + static constexpr const char *NAME = "buildvrt"; + static constexpr const char *DESCRIPTION = "Build a virtual dataset (VRT)."; + static constexpr const char *HELP_URL = + "/programs/gdal_raster_buildvrt.html"; + + static std::vector GetAliases() + { + return {}; + } + + explicit GDALRasterBuildVRTAlgorithm(); + + private: + bool RunImpl(GDALProgressFunc pfnProgress, void *pProgressData) override; + + std::vector m_inputDatasets{}; + GDALArgDatasetValue m_outputDataset{}; + std::vector m_creationOptions{}; + bool m_overwrite = false; + bool m_separate = false; + std::string m_resolution{}; + std::vector m_bbox{}; + bool m_targetAlignedPixels = false; + std::vector m_srcNoData{}; + std::vector m_vrtNoData{}; + std::vector m_bands{}; + bool m_hideNoData = false; + bool m_addAlpha = false; +}; + +//! @endcond + +#endif diff --git a/apps/gdalalg_raster_pipeline.cpp b/apps/gdalalg_raster_pipeline.cpp index c135108959d5..0ab163fa531c 100644 --- a/apps/gdalalg_raster_pipeline.cpp +++ b/apps/gdalalg_raster_pipeline.cpp @@ -17,7 +17,6 @@ #include "gdalalg_raster_write.h" #include "cpl_conv.h" -#include "cpl_json.h" #include "cpl_string.h" #include @@ -149,8 +148,9 @@ bool GDALRasterPipelineStepAlgorithm::RunImpl(GDALProgressFunc pfnProgress, GDALRasterPipelineAlgorithm::GDALRasterPipelineAlgorithm( bool openForMixedRasterVector) - : GDALRasterPipelineStepAlgorithm(NAME, DESCRIPTION, HELP_URL, - /*standaloneStep=*/false) + : GDALAbstractPipelineAlgorithm( + NAME, DESCRIPTION, HELP_URL, + /*standaloneStep=*/false) { AddInputArgs(openForMixedRasterVector, /* hiddenForCLI = */ true); AddProgressArg(); @@ -165,18 +165,6 @@ GDALRasterPipelineAlgorithm::GDALRasterPipelineAlgorithm( m_stepRegistry.Register(); } -/************************************************************************/ -/* GDALRasterPipelineAlgorithm::GetStepAlg() */ -/************************************************************************/ - -std::unique_ptr -GDALRasterPipelineAlgorithm::GetStepAlg(const std::string &name) const -{ - auto alg = m_stepRegistry.Instantiate(name); - return std::unique_ptr( - cpl::down_cast(alg.release())); -} - /************************************************************************/ /* GDALRasterPipelineAlgorithm::ParseCommandLineArguments() */ /************************************************************************/ @@ -387,90 +375,6 @@ bool GDALRasterPipelineAlgorithm::ParseCommandLineArguments( return true; } -/************************************************************************/ -/* GDALRasterPipelineAlgorithm::RunStep() */ -/************************************************************************/ - -bool GDALRasterPipelineAlgorithm::RunStep(GDALProgressFunc pfnProgress, - void *pProgressData) -{ - if (m_steps.empty()) - { - // If invoked programmatically, not from the command line. - - if (m_pipeline.empty()) - { - ReportError(CE_Failure, CPLE_AppDefined, - "'pipeline' argument not set"); - return false; - } - - const CPLStringList aosTokens(CSLTokenizeString(m_pipeline.c_str())); - if (!ParseCommandLineArguments(aosTokens)) - return false; - } - - GDALDataset *poCurDS = nullptr; - for (size_t i = 0; i < m_steps.size(); ++i) - { - auto &step = m_steps[i]; - if (i > 0) - { - if (step->m_inputDataset.GetDatasetRef()) - { - // Shouldn't happen - ReportError(CE_Failure, CPLE_AppDefined, - "Step nr %d (%s) has already an input dataset", - static_cast(i), step->GetName().c_str()); - return false; - } - step->m_inputDataset.Set(poCurDS); - } - if (i + 1 < m_steps.size() && step->m_outputDataset.GetDatasetRef()) - { - // Shouldn't happen - ReportError(CE_Failure, CPLE_AppDefined, - "Step nr %d (%s) has already an output dataset", - static_cast(i), step->GetName().c_str()); - return false; - } - if (!step->Run(i < m_steps.size() - 1 ? nullptr : pfnProgress, - i < m_steps.size() - 1 ? nullptr : pProgressData)) - { - return false; - } - poCurDS = step->m_outputDataset.GetDatasetRef(); - if (!poCurDS) - { - ReportError(CE_Failure, CPLE_AppDefined, - "Step nr %d (%s) failed to produce an output dataset", - static_cast(i), step->GetName().c_str()); - return false; - } - } - - if (!m_outputDataset.GetDatasetRef()) - { - m_outputDataset.Set(poCurDS); - } - - return true; -} - -/************************************************************************/ -/* GDALAlgorithm::Finalize() */ -/************************************************************************/ - -bool GDALRasterPipelineAlgorithm::Finalize() -{ - bool ret = GDALAlgorithm::Finalize(); - for (auto &step : m_steps) - { - ret = step->Finalize() && ret; - } - return ret; -} - /************************************************************************/ /* GDALRasterPipelineAlgorithm::GetUsageForCLI() */ /************************************************************************/ @@ -531,26 +435,4 @@ std::string GDALRasterPipelineAlgorithm::GetUsageForCLI( return ret; } -/************************************************************************/ -/* GDALRasterPipelineAlgorithm::GetUsageAsJSON() */ -/************************************************************************/ - -std::string GDALRasterPipelineAlgorithm::GetUsageAsJSON() const -{ - CPLJSONDocument oDoc; - oDoc.LoadMemory(GDALAlgorithm::GetUsageAsJSON()); - - CPLJSONArray jPipelineSteps; - for (const std::string &name : m_stepRegistry.GetNames()) - { - auto alg = GetStepAlg(name); - CPLJSONDocument oStepDoc; - oStepDoc.LoadMemory(alg->GetUsageAsJSON()); - jPipelineSteps.Add(oStepDoc.GetRoot()); - } - oDoc.GetRoot().Add("pipeline_algorithms", jPipelineSteps); - - return oDoc.SaveAsString(); -} - //! @endcond diff --git a/apps/gdalalg_raster_pipeline.h b/apps/gdalalg_raster_pipeline.h index 0e0a35d86562..7129ac081908 100644 --- a/apps/gdalalg_raster_pipeline.h +++ b/apps/gdalalg_raster_pipeline.h @@ -14,6 +14,7 @@ #define GDALALG_RASTER_PIPELINE_INCLUDED #include "gdalalgorithm.h" +#include "gdalalg_abstract_pipeline.h" //! @cond Doxygen_Suppress @@ -30,6 +31,7 @@ class GDALRasterPipelineStepAlgorithm /* non final */ : public GDALAlgorithm bool standaloneStep); friend class GDALRasterPipelineAlgorithm; + friend class GDALAbstractPipelineAlgorithm; virtual bool RunStep(GDALProgressFunc pfnProgress, void *pProgressData) = 0; @@ -67,7 +69,8 @@ class GDALRasterPipelineStepAlgorithm /* non final */ : public GDALAlgorithm #define GDAL_PIPELINE_PROJ_NOSTALGIA #endif -class GDALRasterPipelineAlgorithm final : public GDALRasterPipelineStepAlgorithm +class GDALRasterPipelineAlgorithm final + : public GDALAbstractPipelineAlgorithm { public: static constexpr const char *NAME = "pipeline"; @@ -91,33 +94,19 @@ class GDALRasterPipelineAlgorithm final : public GDALRasterPipelineStepAlgorithm bool ParseCommandLineArguments(const std::vector &args) override; - bool Finalize() override; - std::string GetUsageForCLI(bool shortUsage, const UsageOptions &usageOptions) const override; - std::string GetUsageAsJSON() const override; - GDALDataset *GetDatasetRef() { return m_inputDataset.GetDatasetRef(); } - /* cppcheck-suppress functionStatic */ - void SetDataset(GDALDataset *) + protected: + GDALArgDatasetValue &GetOutputDataset() override { + return m_outputDataset; } - - private: - std::string m_pipeline{}; - - bool RunStep(GDALProgressFunc pfnProgress, void *pProgressData) override; - - std::unique_ptr - GetStepAlg(const std::string &name) const; - - GDALAlgorithmRegistry m_stepRegistry{}; - std::vector> m_steps{}; }; //! @endcond diff --git a/apps/gdalalg_vector_pipeline.cpp b/apps/gdalalg_vector_pipeline.cpp index 6588ecaf6ad6..7bee6af9a19a 100644 --- a/apps/gdalalg_vector_pipeline.cpp +++ b/apps/gdalalg_vector_pipeline.cpp @@ -17,7 +17,6 @@ #include "gdalalg_vector_write.h" #include "cpl_conv.h" -#include "cpl_json.h" #include "cpl_string.h" #include @@ -161,8 +160,9 @@ bool GDALVectorPipelineStepAlgorithm::RunImpl(GDALProgressFunc pfnProgress, /************************************************************************/ GDALVectorPipelineAlgorithm::GDALVectorPipelineAlgorithm() - : GDALVectorPipelineStepAlgorithm(NAME, DESCRIPTION, HELP_URL, - /*standaloneStep=*/false) + : GDALAbstractPipelineAlgorithm( + NAME, DESCRIPTION, HELP_URL, + /*standaloneStep=*/false) { AddInputArgs(/* hiddenForCLI = */ true); AddProgressArg(); @@ -178,18 +178,6 @@ GDALVectorPipelineAlgorithm::GDALVectorPipelineAlgorithm() m_stepRegistry.Register(); } -/************************************************************************/ -/* GDALVectorPipelineAlgorithm::GetStepAlg() */ -/************************************************************************/ - -std::unique_ptr -GDALVectorPipelineAlgorithm::GetStepAlg(const std::string &name) const -{ - auto alg = m_stepRegistry.Instantiate(name); - return std::unique_ptr( - cpl::down_cast(alg.release())); -} - /************************************************************************/ /* GDALVectorPipelineAlgorithm::ParseCommandLineArguments() */ /************************************************************************/ @@ -444,90 +432,6 @@ bool GDALVectorPipelineAlgorithm::ParseCommandLineArguments( return true; } -/************************************************************************/ -/* GDALVectorPipelineAlgorithm::RunStep() */ -/************************************************************************/ - -bool GDALVectorPipelineAlgorithm::RunStep(GDALProgressFunc pfnProgress, - void *pProgressData) -{ - if (m_steps.empty()) - { - // If invoked programmatically, not from the command line. - - if (m_pipeline.empty()) - { - ReportError(CE_Failure, CPLE_AppDefined, - "'pipeline' argument not set"); - return false; - } - - const CPLStringList aosTokens(CSLTokenizeString(m_pipeline.c_str())); - if (!ParseCommandLineArguments(aosTokens)) - return false; - } - - GDALDataset *poCurDS = nullptr; - for (size_t i = 0; i < m_steps.size(); ++i) - { - auto &step = m_steps[i]; - if (i > 0) - { - if (step->m_inputDataset.GetDatasetRef()) - { - // Shouldn't happen - ReportError(CE_Failure, CPLE_AppDefined, - "Step nr %d (%s) has already an input dataset", - static_cast(i), step->GetName().c_str()); - return false; - } - step->m_inputDataset.Set(poCurDS); - } - if (i + 1 < m_steps.size() && step->m_outputDataset.GetDatasetRef()) - { - // Shouldn't happen - ReportError(CE_Failure, CPLE_AppDefined, - "Step nr %d (%s) has already an output dataset", - static_cast(i), step->GetName().c_str()); - return false; - } - if (!step->Run(i < m_steps.size() - 1 ? nullptr : pfnProgress, - i < m_steps.size() - 1 ? nullptr : pProgressData)) - { - return false; - } - poCurDS = step->m_outputDataset.GetDatasetRef(); - if (!poCurDS) - { - ReportError(CE_Failure, CPLE_AppDefined, - "Step nr %d (%s) failed to produce an output dataset", - static_cast(i), step->GetName().c_str()); - return false; - } - } - - if (!m_outputDataset.GetDatasetRef()) - { - m_outputDataset.Set(poCurDS); - } - - return true; -} - -/************************************************************************/ -/* GDALAlgorithm::Finalize() */ -/************************************************************************/ - -bool GDALVectorPipelineAlgorithm::Finalize() -{ - bool ret = GDALAlgorithm::Finalize(); - for (auto &step : m_steps) - { - ret = step->Finalize() && ret; - } - return ret; -} - /************************************************************************/ /* GDALVectorPipelineAlgorithm::GetUsageForCLI() */ /************************************************************************/ @@ -592,26 +496,4 @@ std::string GDALVectorPipelineAlgorithm::GetUsageForCLI( return ret; } -/************************************************************************/ -/* GDALVectorPipelineAlgorithm::GetUsageAsJSON() */ -/************************************************************************/ - -std::string GDALVectorPipelineAlgorithm::GetUsageAsJSON() const -{ - CPLJSONDocument oDoc; - oDoc.LoadMemory(GDALAlgorithm::GetUsageAsJSON()); - - CPLJSONArray jPipelineSteps; - for (const std::string &name : m_stepRegistry.GetNames()) - { - auto alg = GetStepAlg(name); - CPLJSONDocument oStepDoc; - oStepDoc.LoadMemory(alg->GetUsageAsJSON()); - jPipelineSteps.Add(oStepDoc.GetRoot()); - } - oDoc.GetRoot().Add("pipeline_algorithms", jPipelineSteps); - - return oDoc.SaveAsString(); -} - //! @endcond diff --git a/apps/gdalalg_vector_pipeline.h b/apps/gdalalg_vector_pipeline.h index 14b960dc6485..97564af4e04e 100644 --- a/apps/gdalalg_vector_pipeline.h +++ b/apps/gdalalg_vector_pipeline.h @@ -14,6 +14,7 @@ #define GDALALG_VECTOR_PIPELINE_INCLUDED #include "gdalalgorithm.h" +#include "gdalalg_abstract_pipeline.h" //! @cond Doxygen_Suppress @@ -30,6 +31,7 @@ class GDALVectorPipelineStepAlgorithm /* non final */ : public GDALAlgorithm bool standaloneStep); friend class GDALVectorPipelineAlgorithm; + friend class GDALAbstractPipelineAlgorithm; virtual bool RunStep(GDALProgressFunc pfnProgress, void *pProgressData) = 0; @@ -69,7 +71,8 @@ class GDALVectorPipelineStepAlgorithm /* non final */ : public GDALAlgorithm // "gdal vector pipeline ! read poly.gpkg ! reproject--dst-crs=EPSG:32632 ! write out.gpkg --overwrite" #define GDAL_PIPELINE_PROJ_NOSTALGIA -class GDALVectorPipelineAlgorithm final : public GDALVectorPipelineStepAlgorithm +class GDALVectorPipelineAlgorithm final + : public GDALAbstractPipelineAlgorithm { public: static constexpr const char *NAME = "pipeline"; @@ -93,28 +96,14 @@ class GDALVectorPipelineAlgorithm final : public GDALVectorPipelineStepAlgorithm bool ParseCommandLineArguments(const std::vector &args) override; - bool Finalize() override; - std::string GetUsageForCLI(bool shortUsage, const UsageOptions &usageOptions) const override; - std::string GetUsageAsJSON() const override; - - /* cppcheck-suppress functionStatic */ - void SetDataset(GDALDataset *) + protected: + GDALArgDatasetValue &GetOutputDataset() override { + return m_outputDataset; } - - private: - std::string m_pipeline{}; - - bool RunStep(GDALProgressFunc pfnProgress, void *pProgressData) override; - - std::unique_ptr - GetStepAlg(const std::string &name) const; - - GDALAlgorithmRegistry m_stepRegistry{}; - std::vector> m_steps{}; }; //! @endcond diff --git a/autotest/cpp/test_cpl.cpp b/autotest/cpp/test_cpl.cpp index 038e0fef80d7..5a60bb6bc0e8 100644 --- a/autotest/cpp/test_cpl.cpp +++ b/autotest/cpp/test_cpl.cpp @@ -1334,6 +1334,10 @@ TEST_F(test_cpl, CPLDeclareKnownConfigOption) { CPLDeclareKnownConfigOption("DECLARED_OPTION", nullptr); + const CPLStringList aosKnownConfigOptions(CPLGetKnownConfigOptions()); + EXPECT_GE(aosKnownConfigOptions.FindString("CPL_DEBUG"), 0); + EXPECT_GE(aosKnownConfigOptions.FindString("DECLARED_OPTION"), 0); + CPLErrorStateBackuper oErrorStateBackuper(CPLQuietErrorHandler); CPLErrorReset(); CPLConfigOptionSetter oDeclaredConfigOptionSetter("DECLARED_OPTION", @@ -5615,4 +5619,148 @@ TEST_F(test_cpl, VSIMemGenerateHiddenFilename) EXPECT_TRUE(ENDS_WITH(pszFilename, "/foo.bar")); } } + +TEST_F(test_cpl, VSIGlob) +{ + GByte abyDummyData[1] = {0}; + const std::string osFilenameRadix = VSIMemGenerateHiddenFilename(""); + const std::string osFilename = osFilenameRadix + "trick"; + VSIFCloseL(VSIFileFromMemBuffer(osFilename.c_str(), abyDummyData, + sizeof(abyDummyData), false)); + + { + CPLStringList aosRes( + VSIGlob(osFilename.c_str(), nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(osFilename.substr(0, osFilename.size() - 1).c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 0); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("?rick").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("?rack").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 0); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*ick").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*").c_str(), nullptr, + nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*ack").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 0); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*ic*").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*ac*").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 0); + } + + { + CPLStringList aosRes(VSIGlob( + std::string(osFilenameRadix).append("[st][!s]ic[j-l]").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("[!s]rick").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("[").c_str(), nullptr, + nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 0); + } + + const std::string osFilenameWithSpecialChars = osFilenameRadix + "[!-]"; + VSIFCloseL(VSIFileFromMemBuffer(osFilenameWithSpecialChars.c_str(), + abyDummyData, sizeof(abyDummyData), false)); + + { + CPLStringList aosRes(VSIGlob( + std::string(osFilenameRadix).append("[[][!]a-][-][]]").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilenameWithSpecialChars.c_str()); + } + + const std::string osFilename2 = osFilenameRadix + "truck/track"; + VSIFCloseL(VSIFileFromMemBuffer(osFilename2.c_str(), abyDummyData, + sizeof(abyDummyData), false)); + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*uc*/track").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename2.c_str()); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("*uc*/truck").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 0); + } + + { + CPLStringList aosRes( + VSIGlob(std::string(osFilenameRadix).append("**/track").c_str(), + nullptr, nullptr, nullptr)); + ASSERT_EQ(aosRes.size(), 1); + EXPECT_STREQ(aosRes[0], osFilename2.c_str()); + } + + VSIUnlink(osFilename.c_str()); + VSIUnlink(osFilenameWithSpecialChars.c_str()); + VSIUnlink(osFilename2.c_str()); + VSIUnlink(osFilenameRadix.c_str()); +} + } // namespace diff --git a/autotest/cpp/test_gdal.cpp b/autotest/cpp/test_gdal.cpp index 88a2e9d4a9e3..7ae64a189bad 100644 --- a/autotest/cpp/test_gdal.cpp +++ b/autotest/cpp/test_gdal.cpp @@ -4951,4 +4951,177 @@ TEST_F(test_gdal, GDALComputeRasterMinMaxLocation_with_mask) EXPECT_EQ(nMaxY, 0); } +TEST_F(test_gdal, GDALTranspose2D) +{ + constexpr int COUNT = 6; + const GByte abyData[] = {1, 2, 3, 4, 5, 6}; + GByte abySrcData[COUNT * 2 * sizeof(double)]; + GByte abyDstData[COUNT * 2 * sizeof(double)]; + GByte abyDstAsByteData[COUNT * 2 * sizeof(double)]; + for (int eSrcDTInt = GDT_Byte; eSrcDTInt < GDT_TypeCount; ++eSrcDTInt) + { + const auto eSrcDT = static_cast(eSrcDTInt); + GDALCopyWords(abyData, GDT_Byte, 1, abySrcData, eSrcDT, + GDALGetDataTypeSizeBytes(eSrcDT), COUNT); + for (int eDstDTInt = GDT_Byte; eDstDTInt < GDT_TypeCount; ++eDstDTInt) + { + const auto eDstDT = static_cast(eDstDTInt); + memset(abyDstData, 0, sizeof(abyDstData)); + GDALTranspose2D(abySrcData, eSrcDT, abyDstData, eDstDT, 3, 2); + + memset(abyDstAsByteData, 0, sizeof(abyDstAsByteData)); + GDALCopyWords(abyDstData, eDstDT, GDALGetDataTypeSizeBytes(eDstDT), + abyDstAsByteData, GDT_Byte, 1, COUNT); + + EXPECT_EQ(abyDstAsByteData[0], 1) + << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT; + EXPECT_EQ(abyDstAsByteData[1], 4) + << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT; + EXPECT_EQ(abyDstAsByteData[2], 2) + << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT; + EXPECT_EQ(abyDstAsByteData[3], 5) + << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT; + EXPECT_EQ(abyDstAsByteData[4], 3) + << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT; + EXPECT_EQ(abyDstAsByteData[5], 6) + << "eSrcDT=" << eSrcDT << ", eDstDT=" << eDstDT; + } + } +} + +TEST_F(test_gdal, GDALTranspose2D_Byte_optims) +{ + std::vector in; + for (int i = 0; i < 19 * 17; ++i) + in.push_back(static_cast(i % 256)); + + std::vector out(in.size()); + + // SSSE3 optim (16x16) blocks + { + constexpr int W = 19; + constexpr int H = 17; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // Optim H = 2 with W < 16 + { + constexpr int W = 15; + constexpr int H = 2; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // Optim H = 2 with W >= 16 + { + constexpr int W = 19; + constexpr int H = 2; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // SSSE3 optim H = 3 with W < 16 + { + constexpr int W = 15; + constexpr int H = 3; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // SSSE3 optim H = 3 with W >= 16 + { + constexpr int W = 19; + constexpr int H = 3; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // Optim H = 4 with H < 16 + { + constexpr int W = 15; + constexpr int H = 4; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // Optim H = 4 with H >= 16 + { + constexpr int W = 19; + constexpr int H = 4; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // SSSE3 optim H = 5 with W < 16 + { + constexpr int W = 15; + constexpr int H = 5; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } + + // SSSE3 optim H = 5 with W >= 16 + { + constexpr int W = 19; + constexpr int H = 5; + GDALTranspose2D(in.data(), GDT_Byte, out.data(), GDT_Byte, W, H); + for (int y = 0; y < H; ++y) + { + for (int x = 0; x < W; ++x) + { + EXPECT_EQ(out[x * H + y], in[y * W + x]); + } + } + } +} + } // namespace diff --git a/autotest/cpp/test_gdal_algorithm.cpp b/autotest/cpp/test_gdal_algorithm.cpp index 6174c5826c85..d8ca927ae20c 100644 --- a/autotest/cpp/test_gdal_algorithm.cpp +++ b/autotest/cpp/test_gdal_algorithm.cpp @@ -597,7 +597,13 @@ TEST_F(test_gdal_algorithm, GDALInConstructionAlgorithmArg_AddAlias) MyAlgorithm alg; alg.GetUsageForCLI(false); + EXPECT_NE(alg.GetArg("flag"), nullptr); + EXPECT_NE(alg.GetArg("--flag"), nullptr); + EXPECT_NE(alg.GetArg("-f"), nullptr); + EXPECT_NE(alg.GetArg("f"), nullptr); EXPECT_NE(alg.GetArg("alias"), nullptr); + EXPECT_EQ(alg.GetArg("invalid"), nullptr); + EXPECT_EQ(alg.GetArg("-"), nullptr); } TEST_F(test_gdal_algorithm, GDALInConstructionAlgorithmArg_AddAlias_redundant) @@ -2919,7 +2925,7 @@ TEST_F(test_gdal_algorithm, algorithm_c_api) char **argNames = GDALAlgorithmGetArgNames(hAlg.get()); ASSERT_NE(argNames, nullptr); - EXPECT_EQ(CSLCount(argNames), 12); + EXPECT_EQ(CSLCount(argNames), 13); CSLDestroy(argNames); EXPECT_EQ(GDALAlgorithmGetArg(hAlg.get(), "non_existing"), nullptr); diff --git a/autotest/gcore/data/gtiff/byte_5_bands_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/byte_5_bands_LZW_predictor_2.tif new file mode 100644 index 000000000000..201c6733dc19 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_5_bands_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/byte_DEFLATE.tif b/autotest/gcore/data/gtiff/byte_DEFLATE.tif new file mode 100644 index 000000000000..fa9ec63a6c32 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_DEFLATE.tif differ diff --git a/autotest/gcore/data/gtiff/byte_DEFLATE_tiled.tif b/autotest/gcore/data/gtiff/byte_DEFLATE_tiled.tif new file mode 100644 index 000000000000..c93c3dac502d Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_DEFLATE_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_JPEG.tif b/autotest/gcore/data/gtiff/byte_JPEG.tif new file mode 100644 index 000000000000..8176b1bed6a0 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_JPEG.tif differ diff --git a/autotest/gcore/data/gtiff/byte_JPEG_tiled.tif b/autotest/gcore/data/gtiff/byte_JPEG_tiled.tif new file mode 100644 index 000000000000..4be2fad690a6 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_JPEG_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_JXL.tif b/autotest/gcore/data/gtiff/byte_JXL.tif new file mode 100644 index 000000000000..ff444755fc22 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_JXL.tif differ diff --git a/autotest/gcore/data/gtiff/byte_JXL_tiled.tif b/autotest/gcore/data/gtiff/byte_JXL_tiled.tif new file mode 100644 index 000000000000..7263c74e78ea Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_JXL_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LERC.tif b/autotest/gcore/data/gtiff/byte_LERC.tif new file mode 100644 index 000000000000..6fdc1343eed2 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LERC.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LERC_DEFLATE.tif b/autotest/gcore/data/gtiff/byte_LERC_DEFLATE.tif new file mode 100644 index 000000000000..3e38b0950fc7 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LERC_DEFLATE.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LERC_DEFLATE_tiled.tif b/autotest/gcore/data/gtiff/byte_LERC_DEFLATE_tiled.tif new file mode 100644 index 000000000000..8e8ed7da9b06 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LERC_DEFLATE_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LERC_ZSTD.tif b/autotest/gcore/data/gtiff/byte_LERC_ZSTD.tif new file mode 100644 index 000000000000..2745a5bd5553 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LERC_ZSTD.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LERC_ZSTD_tiled.tif b/autotest/gcore/data/gtiff/byte_LERC_ZSTD_tiled.tif new file mode 100644 index 000000000000..c3636d12e07c Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LERC_ZSTD_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LERC_tiled.tif b/autotest/gcore/data/gtiff/byte_LERC_tiled.tif new file mode 100644 index 000000000000..3fc3978f3955 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LERC_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LZMA.tif b/autotest/gcore/data/gtiff/byte_LZMA.tif new file mode 100644 index 000000000000..9ff97e933efd Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LZMA.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LZMA_tiled.tif b/autotest/gcore/data/gtiff/byte_LZMA_tiled.tif new file mode 100644 index 000000000000..1cafedcb89bd Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LZMA_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LZW.tif b/autotest/gcore/data/gtiff/byte_LZW.tif new file mode 100644 index 000000000000..1232be0405ea Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LZW.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/byte_LZW_predictor_2.tif new file mode 100644 index 000000000000..85ed33ba67db Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/byte_LZW_tiled.tif b/autotest/gcore/data/gtiff/byte_LZW_tiled.tif new file mode 100644 index 000000000000..b43311f0e16a Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_LZW_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_NONE.tif b/autotest/gcore/data/gtiff/byte_NONE.tif new file mode 100644 index 000000000000..a4cbc947432c Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_NONE.tif differ diff --git a/autotest/gcore/data/gtiff/byte_NONE_tiled.tif b/autotest/gcore/data/gtiff/byte_NONE_tiled.tif new file mode 100644 index 000000000000..b75bec93cdeb Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_NONE_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_ZSTD.tif b/autotest/gcore/data/gtiff/byte_ZSTD.tif new file mode 100644 index 000000000000..0a5af3530543 Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_ZSTD.tif differ diff --git a/autotest/gcore/data/gtiff/byte_ZSTD_tiled.tif b/autotest/gcore/data/gtiff/byte_ZSTD_tiled.tif new file mode 100644 index 000000000000..fd8ad1d4538d Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_ZSTD_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/byte_coord_epoch.tif b/autotest/gcore/data/gtiff/byte_coord_epoch.tif new file mode 100644 index 000000000000..89b20114959f Binary files /dev/null and b/autotest/gcore/data/gtiff/byte_coord_epoch.tif differ diff --git a/autotest/gcore/data/gtiff/cint32_big_endian.tif b/autotest/gcore/data/gtiff/cint32_big_endian.tif new file mode 100644 index 000000000000..ed3e7b326a64 Binary files /dev/null and b/autotest/gcore/data/gtiff/cint32_big_endian.tif differ diff --git a/autotest/gcore/data/gtiff/float32_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/float32_LZW_predictor_2.tif new file mode 100644 index 000000000000..77527c94410b Binary files /dev/null and b/autotest/gcore/data/gtiff/float32_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/float32_LZW_predictor_3.tif b/autotest/gcore/data/gtiff/float32_LZW_predictor_3.tif new file mode 100644 index 000000000000..815acce744db Binary files /dev/null and b/autotest/gcore/data/gtiff/float32_LZW_predictor_3.tif differ diff --git a/autotest/gcore/data/gtiff/float64_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/float64_LZW_predictor_2.tif new file mode 100644 index 000000000000..25276fb8ea89 Binary files /dev/null and b/autotest/gcore/data/gtiff/float64_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/float64_LZW_predictor_3.tif b/autotest/gcore/data/gtiff/float64_LZW_predictor_3.tif new file mode 100644 index 000000000000..c3105812eeaf Binary files /dev/null and b/autotest/gcore/data/gtiff/float64_LZW_predictor_3.tif differ diff --git a/autotest/gcore/data/gtiff/generate_test_files.sh b/autotest/gcore/data/gtiff/generate_test_files.sh new file mode 100755 index 000000000000..99c6e0c63c70 --- /dev/null +++ b/autotest/gcore/data/gtiff/generate_test_files.sh @@ -0,0 +1,42 @@ +#!/bin/sh + +for codec in NONE LZW LZMA ZSTD DEFLATE JPEG LERC LERC_DEFLATE LERC_ZSTD JXL +do + gdal_translate ../byte.tif byte_${codec}.tif -co COMPRESS=${codec} + gdal_translate ../byte.tif byte_${codec}_tiled.tif -co COMPRESS=${codec} -co TILED=YES -co BLOCKXSIZE=16 -co BLOCKYSIZE=16 +done + +gdal_translate ../byte.tif byte_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 +gdal_translate ../byte.tif uint16_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt16 +gdal_translate ../byte.tif uint32_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt32 +gdal_translate ../byte.tif uint64_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt64 +gdal_translate ../byte.tif float32_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot Float32 +gdal_translate ../byte.tif float64_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot Float64 +gdal_translate ../byte.tif float32_LZW_predictor_3.tif -co COMPRESS=LZW -co PREDICTOR=3 -ot Float32 +gdal_translate ../byte.tif float64_LZW_predictor_3.tif -co COMPRESS=LZW -co PREDICTOR=3 -ot Float64 + +for codec in NONE LZW LZMA ZSTD DEFLATE JPEG LERC LERC_DEFLATE LERC_ZSTD JXL WEBP +do + gdal_translate ../rgbsmall.tif rgbsmall_${codec}.tif -co COMPRESS=${codec} -co INTERLEAVE=PIXEL -co WEBP_LOSSLESS=YES + gdal_translate ../rgbsmall.tif rgbsmall_${codec}_tiled.tif -co COMPRESS=${codec} -co TILED=YES -co BLOCKXSIZE=16 -co BLOCKYSIZE=32 -co INTERLEAVE=PIXEL -co WEBP_LOSSLESS=YES + if test "${codec}" != "WEBP"; then + gdal_translate ../rgbsmall.tif rgbsmall_${codec}_separate.tif -co COMPRESS=${codec} -co INTERLEAVE=BAND + gdal_translate ../rgbsmall.tif rgbsmall_${codec}_tiled_separate.tif -co COMPRESS=${codec} -co TILED=YES -co BLOCKXSIZE=16 -co BLOCKYSIZE=32 -co INTERLEAVE=BAND + fi +done + +gdal_translate ../rgbsmall.tif rgbsmall_JPEG_ycbcr.tif -co COMPRESS=JPEG -co PHOTOMETRIC=YCBCR -co INTERLEAVE=PIXEL + +gdal_translate ../rgbsmall.tif rgbsmall_byte_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 +gdal_translate ../rgbsmall.tif rgbsmall_uint16_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt16 +gdal_translate ../rgbsmall.tif rgbsmall_uint32_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt32 +gdal_translate ../rgbsmall.tif rgbsmall_uint64_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt64 + +gdal_translate ../stefan_full_rgba.tif stefan_full_rgba_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 + +gdal_translate ../stefan_full_greyalpha.tif stefan_full_greyalpha_byte_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 +gdal_translate ../stefan_full_greyalpha.tif stefan_full_greyalpha_uint16_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt16 +gdal_translate ../stefan_full_greyalpha.tif stefan_full_greyalpha_uint32_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt32 +gdal_translate ../stefan_full_greyalpha.tif stefan_full_greyalpha_uint64_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -ot UInt64 + +gdal_translate ../byte.tif byte_5_bands_LZW_predictor_2.tif -co COMPRESS=LZW -co PREDICTOR=2 -b 1 -b 1 -b 1 -b 1 -b 1 -scale_2 0 255 255 0 -scale_4 0 255 255 0 diff --git a/autotest/gcore/data/gtiff/int16_big_endian.tif b/autotest/gcore/data/gtiff/int16_big_endian.tif new file mode 100644 index 000000000000..3ba26c493cff Binary files /dev/null and b/autotest/gcore/data/gtiff/int16_big_endian.tif differ diff --git a/autotest/gcore/data/gtiff/miniswhite.tif b/autotest/gcore/data/gtiff/miniswhite.tif new file mode 100644 index 000000000000..7eac54b3b847 Binary files /dev/null and b/autotest/gcore/data/gtiff/miniswhite.tif differ diff --git a/autotest/gcore/data/gtiff/non_square_pixels.tif b/autotest/gcore/data/gtiff/non_square_pixels.tif new file mode 100644 index 000000000000..41d49c3d4959 Binary files /dev/null and b/autotest/gcore/data/gtiff/non_square_pixels.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_DEFLATE.tif b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE.tif new file mode 100644 index 000000000000..278b30a63fe6 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_separate.tif new file mode 100644 index 000000000000..8426eeefcb80 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_tiled.tif new file mode 100644 index 000000000000..d4ef80aa7a33 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_tiled_separate.tif new file mode 100644 index 000000000000..7ad8879f820c Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_DEFLATE_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JPEG.tif b/autotest/gcore/data/gtiff/rgbsmall_JPEG.tif new file mode 100644 index 000000000000..33209a3379db Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JPEG.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JPEG_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_JPEG_separate.tif new file mode 100644 index 000000000000..f2a6ccd4fab3 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JPEG_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JPEG_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_JPEG_tiled.tif new file mode 100644 index 000000000000..745fc6bb47c5 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JPEG_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JPEG_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_JPEG_tiled_separate.tif new file mode 100644 index 000000000000..67e8d8cc8edf Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JPEG_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JPEG_ycbcr.tif b/autotest/gcore/data/gtiff/rgbsmall_JPEG_ycbcr.tif new file mode 100644 index 000000000000..c73e683877f6 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JPEG_ycbcr.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JXL.tif b/autotest/gcore/data/gtiff/rgbsmall_JXL.tif new file mode 100644 index 000000000000..20592eca2426 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JXL.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JXL_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_JXL_separate.tif new file mode 100644 index 000000000000..4c59434abf04 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JXL_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JXL_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_JXL_tiled.tif new file mode 100644 index 000000000000..380a80dd3ebc Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JXL_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_JXL_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_JXL_tiled_separate.tif new file mode 100644 index 000000000000..b48437146a20 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_JXL_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC.tif new file mode 100644 index 000000000000..70333807fd13 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE.tif new file mode 100644 index 000000000000..99ab3fe97e52 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_separate.tif new file mode 100644 index 000000000000..5628514f9bd9 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_tiled.tif new file mode 100644 index 000000000000..1ff44d77ca5f Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_tiled_separate.tif new file mode 100644 index 000000000000..0de3b132b3e0 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_DEFLATE_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD.tif new file mode 100644 index 000000000000..13c0bf26a836 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_separate.tif new file mode 100644 index 000000000000..c9ce8569d267 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_tiled.tif new file mode 100644 index 000000000000..84f4de423887 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_tiled_separate.tif new file mode 100644 index 000000000000..91ed277e5e9b Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_ZSTD_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_separate.tif new file mode 100644 index 000000000000..ef09a526195c Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_tiled.tif new file mode 100644 index 000000000000..bbdf43f3a2cd Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LERC_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LERC_tiled_separate.tif new file mode 100644 index 000000000000..388a003ec312 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LERC_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZMA.tif b/autotest/gcore/data/gtiff/rgbsmall_LZMA.tif new file mode 100644 index 000000000000..51986a85893f Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZMA.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZMA_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LZMA_separate.tif new file mode 100644 index 000000000000..92eb3b6ca68d Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZMA_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZMA_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_LZMA_tiled.tif new file mode 100644 index 000000000000..6f683d04606a Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZMA_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZMA_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LZMA_tiled_separate.tif new file mode 100644 index 000000000000..774616590706 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZMA_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZW.tif b/autotest/gcore/data/gtiff/rgbsmall_LZW.tif new file mode 100644 index 000000000000..1eaaa7ca1e6b Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZW.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZW_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LZW_separate.tif new file mode 100644 index 000000000000..fea740d0f2d7 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZW_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZW_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_LZW_tiled.tif new file mode 100644 index 000000000000..29f7795aad8f Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZW_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_LZW_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_LZW_tiled_separate.tif new file mode 100644 index 000000000000..08da747a06ff Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_LZW_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_NONE.tif b/autotest/gcore/data/gtiff/rgbsmall_NONE.tif new file mode 100644 index 000000000000..c44ca9eb7655 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_NONE.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_NONE_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_NONE_separate.tif new file mode 100644 index 000000000000..89b6c969ccde Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_NONE_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_NONE_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_NONE_tiled.tif new file mode 100644 index 000000000000..c4fe461c6008 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_NONE_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_NONE_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_NONE_tiled_separate.tif new file mode 100644 index 000000000000..93b8786e3298 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_NONE_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_WEBP.tif b/autotest/gcore/data/gtiff/rgbsmall_WEBP.tif new file mode 100644 index 000000000000..d2eabd6e508d Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_WEBP.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_WEBP_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_WEBP_tiled.tif new file mode 100644 index 000000000000..32d5d4ecccea Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_WEBP_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_ZSTD.tif b/autotest/gcore/data/gtiff/rgbsmall_ZSTD.tif new file mode 100644 index 000000000000..ca10cae5e758 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_ZSTD.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_ZSTD_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_ZSTD_separate.tif new file mode 100644 index 000000000000..b43d103005fe Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_ZSTD_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_ZSTD_tiled.tif b/autotest/gcore/data/gtiff/rgbsmall_ZSTD_tiled.tif new file mode 100644 index 000000000000..b0ad9e074836 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_ZSTD_tiled.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_ZSTD_tiled_separate.tif b/autotest/gcore/data/gtiff/rgbsmall_ZSTD_tiled_separate.tif new file mode 100644 index 000000000000..2cd4bb597f22 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_ZSTD_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_byte_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/rgbsmall_byte_LZW_predictor_2.tif new file mode 100644 index 000000000000..8f9e1ab953e9 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_byte_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_int16_bigendian_lzw_predictor_2.tif b/autotest/gcore/data/gtiff/rgbsmall_int16_bigendian_lzw_predictor_2.tif new file mode 100644 index 000000000000..a34e9d386f52 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_int16_bigendian_lzw_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_uint16_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/rgbsmall_uint16_LZW_predictor_2.tif new file mode 100644 index 000000000000..0f1f48ed8f1f Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_uint16_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_uint32_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/rgbsmall_uint32_LZW_predictor_2.tif new file mode 100644 index 000000000000..e54a5a91ce01 Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_uint32_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/rgbsmall_uint64_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/rgbsmall_uint64_LZW_predictor_2.tif new file mode 100644 index 000000000000..89506254f8df Binary files /dev/null and b/autotest/gcore/data/gtiff/rgbsmall_uint64_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/sparse_nodata_one.tif b/autotest/gcore/data/gtiff/sparse_nodata_one.tif new file mode 100644 index 000000000000..cf7f3acd65dd Binary files /dev/null and b/autotest/gcore/data/gtiff/sparse_nodata_one.tif differ diff --git a/autotest/gcore/data/gtiff/sparse_tiled_contig.tif b/autotest/gcore/data/gtiff/sparse_tiled_contig.tif new file mode 100644 index 000000000000..65558a65b933 Binary files /dev/null and b/autotest/gcore/data/gtiff/sparse_tiled_contig.tif differ diff --git a/autotest/gcore/data/gtiff/sparse_tiled_separate.tif b/autotest/gcore/data/gtiff/sparse_tiled_separate.tif new file mode 100644 index 000000000000..42672bd1a55d Binary files /dev/null and b/autotest/gcore/data/gtiff/sparse_tiled_separate.tif differ diff --git a/autotest/gcore/data/gtiff/stefan_full_greyalpha_byte_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/stefan_full_greyalpha_byte_LZW_predictor_2.tif new file mode 100644 index 000000000000..79747260c2e2 Binary files /dev/null and b/autotest/gcore/data/gtiff/stefan_full_greyalpha_byte_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint16_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint16_LZW_predictor_2.tif new file mode 100644 index 000000000000..f16e26e57f99 Binary files /dev/null and b/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint16_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint32_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint32_LZW_predictor_2.tif new file mode 100644 index 000000000000..de64da93b225 Binary files /dev/null and b/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint32_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint64_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint64_LZW_predictor_2.tif new file mode 100644 index 000000000000..6e1e476a85fd Binary files /dev/null and b/autotest/gcore/data/gtiff/stefan_full_greyalpha_uint64_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/stefan_full_rgba_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/stefan_full_rgba_LZW_predictor_2.tif new file mode 100644 index 000000000000..2e8d2fc2a50b Binary files /dev/null and b/autotest/gcore/data/gtiff/stefan_full_rgba_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/uint16_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/uint16_LZW_predictor_2.tif new file mode 100644 index 000000000000..b2b630ed7253 Binary files /dev/null and b/autotest/gcore/data/gtiff/uint16_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/uint32_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/uint32_LZW_predictor_2.tif new file mode 100644 index 000000000000..0f667d0cb56b Binary files /dev/null and b/autotest/gcore/data/gtiff/uint32_LZW_predictor_2.tif differ diff --git a/autotest/gcore/data/gtiff/uint64_LZW_predictor_2.tif b/autotest/gcore/data/gtiff/uint64_LZW_predictor_2.tif new file mode 100644 index 000000000000..c3540d5ed748 Binary files /dev/null and b/autotest/gcore/data/gtiff/uint64_LZW_predictor_2.tif differ diff --git a/autotest/gcore/libertiff.py b/autotest/gcore/libertiff.py new file mode 100755 index 000000000000..6d8b535dfcd0 --- /dev/null +++ b/autotest/gcore/libertiff.py @@ -0,0 +1,861 @@ +#!/usr/bin/env pytest +# -*- coding: utf-8 -*- +############################################################################### +# Project: GDAL/OGR Test Suite +# Purpose: Test LIBERTIFF driver +# Author: Even Rouault +# +############################################################################### +# Copyright (c) 2024, Even Rouault +# +# SPDX-License-Identifier: MIT +############################################################################### + +import glob +import math +import os +import threading + +import gdaltest +import pytest + +from osgeo import gdal + +pytestmark = pytest.mark.require_driver("LIBERTIFF") + + +def libertiff_open(filename, open_options=[]): + return gdal.OpenEx( + filename, allowed_drivers=["LIBERTIFF"], open_options=open_options + ) + + +def test_libertiff_basic(): + ds = libertiff_open("data/byte.tif") + assert ds.GetMetadataItem("AREA_OR_POINT") == "Area" + assert ds.GetSpatialRef().GetAuthorityCode(None) == "26711" + assert ds.GetGeoTransform() == pytest.approx( + (440720.0, 60.0, 0.0, 3751320.0, 0.0, -60.0) + ) + assert ds.GetGCPCount() == 0 + assert ds.GetGCPSpatialRef() is None + assert len(ds.GetGCPs()) == 0 + assert ds.GetRasterBand(1).GetColorInterpretation() == gdal.GCI_GrayIndex + assert ds.GetRasterBand(1).GetColorTable() is None + assert ds.GetRasterBand(1).GetMaskFlags() == gdal.GMF_ALL_VALID + assert ds.GetRasterBand(1).GetMaskBand().ComputeRasterMinMax() == (255, 255) + assert ds.GetRasterBand(1).GetNoDataValue() is None + assert ds.GetRasterBand(1).GetOffset() is None + assert ds.GetRasterBand(1).GetScale() is None + assert ds.GetRasterBand(1).GetDescription() == "" + assert ds.GetRasterBand(1).GetUnitType() == "" + assert ds.GetRasterBand(1).GetMetadataItem("JPEGTABLES", "TIFF") is None + assert ds.GetRasterBand(1).GetMetadataItem("IFD_OFFSET", "TIFF") == "408" + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_OFFSET_0_0", "TIFF") == "8" + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_OFFSET_0_1", "TIFF") is None + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_OFFSET_1_0", "TIFF") is None + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_SIZE_0_0", "TIFF") == "400" + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_SIZE_0_1", "TIFF") is None + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_SIZE_1_0", "TIFF") is None + assert ( + ds.GetRasterBand(1).InterpolateAtPoint(0, 0, gdal.GRIORA_NearestNeighbour) + == 107.0 + ) + with pytest.raises(Exception): + ds.GetRasterBand(1).Fill(0) + + +@pytest.mark.require_driver("GTiff") +def test_libertiff_basic_compare_gtiff(): + ds = libertiff_open("data/byte.tif") + ds_ref = gdal.OpenEx("data/byte.tif", allowed_drivers=["GTiff"]) + assert ds.ReadRaster() == ds_ref.ReadRaster() + assert ds.ReadRaster(1, 2, 1, 1) == ds_ref.ReadRaster(1, 2, 1, 1) + assert ds.ReadRaster(band_list=[1, 1]) == ds_ref.ReadRaster(band_list=[1, 1]) + assert ds.ReadRaster(1, 2, 3, 4) == ds_ref.ReadRaster(1, 2, 3, 4) + assert ds.ReadRaster(buf_type=gdal.GDT_Int16) == ds_ref.ReadRaster( + buf_type=gdal.GDT_Int16 + ) + assert ds.ReadRaster(buf_xsize=3, buf_ysize=5) == ds_ref.ReadRaster( + buf_xsize=3, buf_ysize=5 + ) + assert ds.ReadRaster( + buf_xsize=3, buf_ysize=5, resample_alg=gdal.GRIORA_Bilinear + ) == ds_ref.ReadRaster(buf_xsize=3, buf_ysize=5, resample_alg=gdal.GRIORA_Bilinear) + + +@pytest.mark.require_driver("GTiff") +@pytest.mark.parametrize("GDAL_FORCE_CACHING", ["YES", "NO"]) +def test_libertiff_3band_separate(GDAL_FORCE_CACHING): + with gdal.config_option("GDAL_FORCE_CACHING", GDAL_FORCE_CACHING): + ds = libertiff_open("data/rgbsmall.tif") + ds_ref = gdal.OpenEx("data/rgbsmall.tif", allowed_drivers=["GTiff"]) + assert ds.ReadRaster() == ds_ref.ReadRaster() + assert ds.GetRasterBand(2).ReadRaster() == ds_ref.GetRasterBand(2).ReadRaster() + assert ds.ReadRaster(1, 2, 1, 1) == ds_ref.ReadRaster(1, 2, 1, 1) + assert ds.ReadRaster(band_list=[3, 2, 1]) == ds_ref.ReadRaster(band_list=[3, 2, 1]) + assert ds.ReadRaster(1, 2, 3, 4) == ds_ref.ReadRaster(1, 2, 3, 4) + assert ds.ReadRaster(buf_type=gdal.GDT_Int16) == ds_ref.ReadRaster( + buf_type=gdal.GDT_Int16 + ) + assert ds.ReadRaster(buf_xsize=3, buf_ysize=5) == ds_ref.ReadRaster( + buf_xsize=3, buf_ysize=5 + ) + assert ds.ReadRaster( + 1.5, 2.5, 6.5, 7.5, buf_xsize=3, buf_ysize=5, resample_alg=gdal.GRIORA_Bilinear + ) == ds_ref.ReadRaster( + 1.5, 2.5, 6.5, 7.5, buf_xsize=3, buf_ysize=5, resample_alg=gdal.GRIORA_Bilinear + ) + + +@pytest.mark.require_driver("GTiff") +@pytest.mark.parametrize("GDAL_FORCE_CACHING", ["YES", "NO"]) +def test_libertiff_4band_pixel_interleaved(GDAL_FORCE_CACHING): + with gdal.config_option("GDAL_FORCE_CACHING", GDAL_FORCE_CACHING): + ds = libertiff_open("data/stefan_full_rgba.tif") + ds_ref = gdal.OpenEx("data/stefan_full_rgba.tif", allowed_drivers=["GTiff"]) + assert ds.ReadRaster() == ds_ref.ReadRaster() + assert ds.ReadRaster(buf_pixel_space=4, buf_band_space=1) == ds_ref.ReadRaster( + buf_pixel_space=4, buf_band_space=1 + ) + assert ds.ReadRaster(band_list=[3, 2, 1, 4]) == ds_ref.ReadRaster( + band_list=[3, 2, 1, 4] + ) + assert ds.ReadRaster(band_list=[3, 2, 1]) == ds_ref.ReadRaster(band_list=[3, 2, 1]) + + +def test_libertiff_with_ovr(): + ds = libertiff_open("data/byte_with_ovr.tif") + assert ds.GetRasterBand(1).GetOverviewCount() == 2 + assert ds.GetRasterBand(1).GetOverview(-1) is None + assert ds.GetRasterBand(1).GetOverview(2) is None + assert ds.GetRasterBand(1).GetOverview(0).Checksum() == 1087 + assert ds.GetRasterBand(1).GetOverview(1).Checksum() == 328 + assert ( + ds.GetRasterBand(1).ReadRaster(buf_xsize=10, buf_ysize=10) + == ds.GetRasterBand(1).GetOverview(0).ReadRaster() + ) + + +@pytest.mark.require_driver("JPEG") +def test_libertiff_with_mask(): + ds = libertiff_open("data/ycbcr_with_mask.tif") + assert ds.GetSpatialRef() is not None + assert ds.RasterCount == 3 + assert ds.GetRasterBand(1).GetMaskFlags() == gdal.GMF_PER_DATASET + assert ds.GetRasterBand(1).GetMaskBand().Checksum() == 1023 + assert ds.GetRasterBand(1).GetMetadataItem("JPEGTABLES", "TIFF") is not None + + +def test_libertiff_nodata(): + ds = libertiff_open("data/nan32_nodata.tif") + assert math.isnan(ds.GetRasterBand(1).GetNoDataValue()) + + +def test_libertiff_sparse_nodata_one(): + ds = libertiff_open("data/gtiff/sparse_nodata_one.tif") + assert ds.GetRasterBand(1).GetNoDataValue() == 1 + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_OFFSET_0_0", "TIFF") is None + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_SIZE_0_0", "TIFF") is None + assert ds.GetRasterBand(1).Checksum() == 1 + assert ds.GetRasterBand(1).ReadRaster(0, 0, 1, 1) == b"\x01" + assert ds.GetRasterBand(1).ReadRaster(0, 0, 1, 1) == b"\x01" + + +@pytest.mark.parametrize("filename", ["sparse_tiled_separate", "sparse_tiled_contig"]) +def test_libertiff_sparse_tiled(filename): + ds = libertiff_open(f"data/gtiff/{filename}.tif") + assert ds.GetRasterBand(1).ReadRaster() == b"\x01" * (40 * 21) + assert ds.GetRasterBand(1).ReadRaster(0, 0, 1, 1) == b"\x01" + + +def test_libertiff_metadata(): + ds = libertiff_open("data/complex_float32.tif") + assert ds.GetMetadata() == { + "bands": "1", + "byte_order": "0", + "data_type": "6", + "description": "File Imported into ENVI.", + "file_type": "ENVI Standard", + "header_offset": "0", + "interleave": "bsq", + "lines": "4148", + "samples": "9400", + "sensor_type": "Unknown", + "wavelength_units": "Unknown", + } + + +def test_libertiff_check_signed_int8(): + ds = libertiff_open("../gdrivers/data/gtiff/int8.tif") + assert ds.GetRasterBand(1).Checksum() == 1046 + + +@pytest.mark.parametrize( + "filename", ["int16.tif", "gtiff/int16_big_endian.tif", "int32.tif", "int64.tif"] +) +def test_libertiff_check_signed_int(filename): + ds = libertiff_open("data/" + filename) + assert ds.GetRasterBand(1).Checksum() == 4672 + + +@pytest.mark.parametrize( + "filename", + [ + "cint16.tif", + "cint32.tif", + "gtiff/cint32_big_endian.tif", + "cfloat32.tif", + "cfloat64.tif", + ], +) +def test_libertiff_check_complex(filename): + ds = libertiff_open("data/" + filename) + assert ds.GetRasterBand(1).Checksum() == 5028 + + +@pytest.mark.parametrize( + "filename_middle", + [ + "DEFLATE", + "DEFLATE_tiled", + "JPEG", + "JPEG_tiled", + "JXL", + "JXL_tiled", + "LERC_DEFLATE", + "LERC_DEFLATE_tiled", + "LERC", + "LERC_tiled", + "LERC_ZSTD", + "LERC_ZSTD_tiled", + "little_endian_golden", + "little_endian_tiled_lzw_golden", + "LZMA", + "LZMA_tiled", + "LZW_predictor_2", + "LZW", + "LZW_tiled", + "NONE", + "NONE_tiled", + "ZSTD", + "ZSTD_tiled", + ], +) +def test_libertiff_check_byte(filename_middle): + if "JPEG" in filename_middle and gdal.GetDriverByName("JPEG") is None: + pytest.skip("JPEG driver missing") + if "JXL" in filename_middle and gdal.GetDriverByName("JPEGXL") is None: + pytest.skip("JPEGXL driver missing") + if "WEBP" in filename_middle and gdal.GetDriverByName("WEBP") is None: + pytest.skip("WEBP driver missing") + if ( + "LZMA" in filename_middle + and gdal.GetDriverByName("LIBERTIFF").GetMetadataItem( + "LZMA_SUPPORT", "LIBERTIFF" + ) + is None + ): + pytest.skip("LZMA support missing") + if ( + "LERC" in filename_middle + and gdal.GetDriverByName("LIBERTIFF").GetMetadataItem( + "LERC_SUPPORT", "LIBERTIFF" + ) + is None + ): + pytest.skip("LERC support missing") + if ( + "ZSTD" in filename_middle + and gdal.GetDriverByName("LIBERTIFF").GetMetadataItem( + "ZSTD_SUPPORT", "LIBERTIFF" + ) + is None + ): + pytest.skip("ZSTD support missing") + + ds = libertiff_open(f"data/gtiff/byte_{filename_middle}.tif") + if "JPEG" in filename_middle: + assert ds.GetRasterBand(1).Checksum() > 0 + if gdal.GetDriverByName("GTiff"): + ds_ref = gdal.OpenEx( + f"data/gtiff/byte_{filename_middle}.tif", allowed_drivers=["GTiff"] + ) + assert ds.GetRasterBand(1).Checksum() == ds_ref.GetRasterBand(1).Checksum() + else: + assert ds.GetRasterBand(1).Checksum() == 4672 + + +@pytest.mark.parametrize( + "filename_middle", + [ + "byte_LZW_predictor_2", + "DEFLATE_separate", + "DEFLATE", + "DEFLATE_tiled_separate", + "DEFLATE_tiled", + "JPEG_separate", + "JPEG", + "JPEG_tiled_separate", + "JPEG_tiled", + "JPEG_ycbcr", + "JXL_separate", + "JXL", + "JXL_tiled_separate", + "JXL_tiled", + "LERC_DEFLATE_separate", + "LERC_DEFLATE", + "LERC_DEFLATE_tiled_separate", + "LERC_DEFLATE_tiled", + "LERC_separate", + "LERC", + "LERC_tiled_separate", + "LERC_tiled", + "LERC_ZSTD_separate", + "LERC_ZSTD", + "LERC_ZSTD_tiled_separate", + "LERC_ZSTD_tiled", + "LZMA_separate", + "LZMA", + "LZMA_tiled_separate", + "LZMA_tiled", + "LZW_separate", + "LZW", + "LZW_tiled_separate", + "LZW_tiled", + "NONE_separate", + "NONE", + "NONE_tiled_separate", + "NONE_tiled", + "uint16_LZW_predictor_2", + "uint32_LZW_predictor_2", + "uint64_LZW_predictor_2", + "WEBP", + "WEBP_tiled", + "ZSTD_separate", + "ZSTD", + "ZSTD_tiled_separate", + "ZSTD_tiled", + "int16_bigendian_lzw_predictor_2", + ], +) +@pytest.mark.parametrize("NUM_THREADS", [None, "2", "ALL_CPUS"]) +def test_libertiff_check_rgbsmall(filename_middle, NUM_THREADS): + if "JPEG" in filename_middle and gdal.GetDriverByName("JPEG") is None: + pytest.skip("JPEG driver missing") + if "JXL" in filename_middle and gdal.GetDriverByName("JPEGXL") is None: + pytest.skip("JPEGXL driver missing") + if "WEBP" in filename_middle and gdal.GetDriverByName("WEBP") is None: + pytest.skip("WEBP driver missing") + if ( + "LZMA" in filename_middle + and gdal.GetDriverByName("LIBERTIFF").GetMetadataItem( + "LZMA_SUPPORT", "LIBERTIFF" + ) + is None + ): + pytest.skip("LZMA support missing") + if ( + "LERC" in filename_middle + and gdal.GetDriverByName("LIBERTIFF").GetMetadataItem( + "LERC_SUPPORT", "LIBERTIFF" + ) + is None + ): + pytest.skip("LERC support missing") + if ( + "ZSTD" in filename_middle + and gdal.GetDriverByName("LIBERTIFF").GetMetadataItem( + "ZSTD_SUPPORT", "LIBERTIFF" + ) + is None + ): + pytest.skip("ZSTD support missing") + + ds = libertiff_open( + f"data/gtiff/rgbsmall_{filename_middle}.tif", + open_options=([] if NUM_THREADS is None else ["NUM_THREADS=" + NUM_THREADS]), + ) + if "JPEG" in filename_middle: + assert ds.GetRasterBand(1).Checksum() > 0 + if gdal.GetDriverByName("GTiff"): + ds_ref = gdal.OpenEx( + f"data/gtiff/rgbsmall_{filename_middle}.tif", + allowed_drivers=["GTiff"], + ) + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(3)] == [ + ds_ref.GetRasterBand(i + 1).Checksum() for i in range(3) + ] + else: + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(3)] == [ + 21212, + 21053, + 21349, + ] + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_OFFSET_0_0", "TIFF") is not None + assert ds.GetRasterBand(1).GetMetadataItem("BLOCK_SIZE_0_0", "TIFF") is not None + + +@pytest.mark.parametrize( + "filename_prefix", + [ + "byte_LZW_predictor_2", + "float32_LZW_predictor_2", + "float32_LZW_predictor_3", + "float32_lzw_predictor_3_big_endian", + "float64_LZW_predictor_2", + "float64_LZW_predictor_3", + "uint16_LZW_predictor_2", + "uint32_LZW_predictor_2", + "uint64_LZW_predictor_2", + ], +) +def test_libertiff_check_predictor_1_band(filename_prefix): + ds = libertiff_open(f"data/gtiff/{filename_prefix}.tif") + assert ds.GetRasterBand(1).Checksum() == 4672 + + +@pytest.mark.parametrize( + "filename_prefix", + [ + "stefan_full_greyalpha_byte_LZW_predictor_2", + "stefan_full_greyalpha_uint16_LZW_predictor_2", + "stefan_full_greyalpha_uint32_LZW_predictor_2", + "stefan_full_greyalpha_uint64_LZW_predictor_2", + ], +) +def test_libertiff_check_predictor_2_bands(filename_prefix): + ds = libertiff_open(f"data/gtiff/{filename_prefix}.tif") + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(2)] == [1970, 10807] + + +@pytest.mark.parametrize( + "filename_prefix", + [ + "rgbsmall_byte_LZW_predictor_2", + "rgbsmall_uint16_LZW_predictor_2", + "rgbsmall_uint32_LZW_predictor_2", + "rgbsmall_uint64_LZW_predictor_2", + ], +) +def test_libertiff_check_predictor_3_bands(filename_prefix): + ds = libertiff_open(f"data/gtiff/{filename_prefix}.tif") + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(3)] == [ + 21212, + 21053, + 21349, + ] + + +def test_libertiff_read_predictor_4_bands(): + ds = libertiff_open("data/gtiff/stefan_full_rgba_LZW_predictor_2.tif") + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(4)] == [ + 12603, + 58561, + 36064, + 10807, + ] + + +def test_libertiff_read_predictor_5_bands(): + ds = libertiff_open("data/gtiff/byte_5_bands_LZW_predictor_2.tif") + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(5)] == [ + 4672, + 4563, + 4672, + 4563, + 4672, + ] + + +############################################################################### +# Test reading a GeoTIFF file with tiepoints in PixelIsPoint format. + + +def test_libertiff_read_tiepoints_pixelispoint(): + + ds = libertiff_open("data/byte_gcp_pixelispoint.tif") + assert ds.GetMetadataItem("AREA_OR_POINT") == "Point" + assert ds.GetSpatialRef() is None + assert ds.GetGCPSpatialRef() is not None + assert ds.GetGCPCount() == 4 + gcp = ds.GetGCPs()[0] + assert ( + gcp.GCPPixel == pytest.approx(0.5, abs=1e-5) + and gcp.GCPLine == pytest.approx(0.5, abs=1e-5) + and gcp.GCPX == pytest.approx(-180, abs=1e-5) + and gcp.GCPY == pytest.approx(90, abs=1e-5) + and gcp.GCPZ == pytest.approx(0, abs=1e-5) + ) + + +############################################################################### +# Test reading a 4-band contig jpeg compressed geotiff. + + +@pytest.mark.require_driver("JPEG") +def test_libertiff_rgba_jpeg_contig(): + + ds = libertiff_open("data/stefan_full_rgba_jpeg_contig.tif") + assert ds.GetRasterBand(1).Checksum() > 0 + assert ds.GetRasterBand(2).Checksum() > 0 + assert ds.GetRasterBand(3).Checksum() > 0 + assert ds.GetRasterBand(4).Checksum() > 0 + + +############################################################################### +# Test reading a 12bit jpeg compressed geotiff. + + +@pytest.mark.skipif( + "SKIP_TIFF_JPEG12" in os.environ, reason="Crashes on build-windows-msys2-mingw" +) +@pytest.mark.require_driver("JPEG") +def test_libertiff_12bitjpeg(tmp_vsimem): + + jpeg_drv = gdal.GetDriverByName("JPEG") + if "UInt16" not in jpeg_drv.GetMetadataItem(gdal.DMD_CREATIONDATATYPES): + pytest.skip("12-bit JPEG not supported in this build") + + filename = str(tmp_vsimem / "tmp.tif") + gdal.FileFromMemBuffer( + filename, open("data/mandrilmini_12bitjpeg.tif", "rb").read() + ) + ds = libertiff_open(filename) + + stats = ds.GetRasterBand(1).ComputeStatistics(False) + assert not ( + stats[2] < 2150 or stats[2] > 2180 or str(stats[2]) == "nan" + ), "did not get expected mean for band1." + + +############################################################################### + + +@pytest.mark.parametrize( + "filename", ["data/excessive-memory-TIFFFillStrip.tif", "data/huge4GB.tif"] +) +def test_libertiff_read_error(filename): + + ds = libertiff_open(filename) + for i in range(ds.RasterCount): + with pytest.raises(Exception): + ds.GetRasterBand(i + 1).Checksum() + + +############################################################################### + + +@pytest.mark.parametrize( + "filename", + [ + "data/huge-number-strips.tiff", + ], +) +def test_libertiff_open_error(filename): + + with pytest.raises(Exception): + libertiff_open(filename) + + +############################################################################### +# Test we don't crash on files + + +@gdaltest.disable_exceptions() +def test_libertiff_test_all_tif(): + + for filename in glob.glob("data/*.tif"): + # print(filename) + with gdal.quiet_errors(): + ds = libertiff_open(filename) + if ds and ds.RasterCount: + ds.ReadRaster(0, 0, 1, 1) + + +def test_libertiff_thread_safe_readraster(): + + ds = libertiff_open("data/gtiff/rgbsmall_DEFLATE_tiled_separate.tif") + assert ds.IsThreadSafe(gdal.OF_RASTER) + + res = [True] + + def check(): + for i in range(100): + if [ds.GetRasterBand(i + 1).Checksum() for i in range(3)] != [ + 21212, + 21053, + 21349, + ]: + res[0] = False + + threads = [threading.Thread(target=check) for i in range(2)] + for t in threads: + t.start() + for t in threads: + t.join() + assert res[0] + + +############################################################################### +# Test different datatypes for StripOffsets tag with little/big, classic/bigtiff + + +@pytest.mark.parametrize( + "filename,expected_offsets", + [ + ("data/classictiff_one_block_byte.tif", [158]), + ("data/classictiff_one_block_long.tif", [158]), + ("data/classictiff_one_block_be_long.tif", [158]), + ("data/classictiff_one_strip_long.tif", [146]), + ("data/classictiff_one_strip_be_long.tif", [146]), + ("data/classictiff_two_strip_short.tif", [162, 163]), + ("data/classictiff_two_strip_be_short.tif", [162, 163]), + ("data/classictiff_four_strip_short.tif", [178, 179, 180, 181]), + ("data/classictiff_four_strip_be_short.tif", [178, 179, 180, 181]), + ("data/bigtiff_four_strip_short.tif", [316, 317, 318, 319]), + ("data/bigtiff_four_strip_be_short.tif", [316, 317, 318, 319]), + ("data/bigtiff_one_block_long8.tif", [272]), + ("data/bigtiff_one_block_be_long8.tif", [272]), + ("data/bigtiff_one_strip_long.tif", [252]), + ("data/bigtiff_one_strip_be_long.tif", [252]), + ("data/bigtiff_one_strip_long8.tif", [252]), + ("data/bigtiff_one_strip_be_long8.tif", [252]), + ("data/bigtiff_two_strip_long.tif", [284, 285]), + ("data/bigtiff_two_strip_be_long.tif", [284, 285]), + ("data/bigtiff_two_strip_long8.tif", [284, 285]), + ("data/bigtiff_two_strip_be_long8.tif", [284, 285]), + ], +) +def test_libertiff_read_stripoffset_types(filename, expected_offsets): + + ds = libertiff_open(filename) + offsets = [] + for row in range(4): + mdi = ds.GetRasterBand(1).GetMetadataItem("BLOCK_OFFSET_0_%d" % row, "TIFF") + if mdi is None: + break + offsets.append(int(mdi)) + assert offsets == expected_offsets + assert ds.GetRasterBand(1).Checksum() > 0 + + +def test_libertiff_packbits(): + ds = libertiff_open("data/seperate_strip.tif") + assert [ds.GetRasterBand(i + 1).Checksum() for i in range(3)] == [ + 15234, + 15234, + 15234, + ] + + +def test_libertiff_coordinate_epoch(): + ds = libertiff_open("data/gtiff/byte_coord_epoch.tif") + assert ds.GetSpatialRef().GetCoordinateEpoch() == 2020.0 + + +@pytest.mark.require_driver("WEBP") +def test_libertiff_webp_lossy(): + ds = libertiff_open("data/tif_webp.tif") + assert ds.GetMetadata("IMAGE_STRUCTURE") == { + "COMPRESSION_REVERSIBILITY": "LOSSY", + "COMPRESSION": "WEBP", + "INTERLEAVE": "PIXEL", + } + + +def test_libertiff_multi_image(): + ds = libertiff_open("data/twoimages.tif") + assert ds.RasterCount == 0 + assert ds.GetSubDatasets() == [ + ("GTIFF_DIR:1:data/twoimages.tif", "Page 1 (20P x 20L x 1B)"), + ("GTIFF_DIR:2:data/twoimages.tif", "Page 2 (20P x 20L x 1B)"), + ] + + with pytest.raises(Exception): + libertiff_open("GTIFF_DIR:") + with pytest.raises(Exception): + libertiff_open("GTIFF_DIR:0") + with pytest.raises(Exception): + libertiff_open("GTIFF_DIR:-1:data/twoimages.tif") + with pytest.raises(Exception): + libertiff_open("GTIFF_DIR:0:data/twoimages.tif") + with pytest.raises(Exception): + libertiff_open("GTIFF_DIR:3:data/twoimages.tif") + with pytest.raises(Exception): + libertiff_open("GTIFF_DIR:1:i_do_not_exist.tif") + + ds = libertiff_open("GTIFF_DIR:1:data/twoimages.tif") + assert ds.GetRasterBand(1).GetMetadataItem("IFD_OFFSET", "TIFF") == "408" + + ds = libertiff_open("GTIFF_DIR:2:data/twoimages.tif") + assert ds.GetRasterBand(1).GetMetadataItem("IFD_OFFSET", "TIFF") == "958" + + +def test_libertiff_miniswhite(): + ds = libertiff_open("data/gtiff/miniswhite.tif") + assert ds.GetMetadataItem("MINISWHITE", "IMAGE_STRUCTURE") == "YES" + assert ds.GetRasterBand(1).GetColorInterpretation() == gdal.GCI_PaletteIndex + ct = ds.GetRasterBand(1).GetColorTable() + assert ct + assert ct.GetCount() == 2 + assert ct.GetColorEntry(0) == (255, 255, 255, 255) + assert ct.GetColorEntry(1) == (0, 0, 0, 255) + + +def test_libertiff_colortable(): + ds = libertiff_open("data/test_average_palette.tif") + assert ds.GetRasterBand(1).GetColorInterpretation() == gdal.GCI_PaletteIndex + ct = ds.GetRasterBand(1).GetColorTable() + assert ct + assert ct.GetCount() == 256 + assert ct.GetColorEntry(0) == (0, 0, 0, 255) + assert ct.GetColorEntry(1) == (255, 255, 255, 255) + assert ct.GetColorEntry(2) == (127, 127, 127, 255) + + +############################################################################### +# Test reading a TIFF with the RPC tag per +# http://geotiff.maptools.org/rpc_prop.html + + +def test_libertiff_read_rpc_tif(): + + ds = libertiff_open("data/byte_rpc.tif") + got_md = ds.GetMetadata("RPC") + expected_md = { + "ERR_BIAS": "-1", + "ERR_RAND": "-1", + "HEIGHT_OFF": "300", + "HEIGHT_SCALE": "970", + "LAT_OFF": "-42.8607", + "LAT_SCALE": "0.0715", + "LINE_DEN_COEFF": "1 -0.00520769693945429 0.00549138470185778 " + "-0.00472718007439368 -2.02057365543208e-05 " + "8.39509604990839e-06 -3.43728317291176e-05 " + "-1.94598141718645e-05 2.35660616018416e-05 " + "3.71653924834455e-06 7.08406853351595e-08 " + "-5.72002373463043e-09 8.71363385561163e-09 " + "-4.39082343385483e-09 1.39161295297537e-08 " + "-1.24914964124471e-08 4.6171143198849e-08 " + "1.70067569381897e-08 -2.37449991316111e-08 " + "9.05460084990073e-10", + "LINE_NUM_COEFF": "-0.000539636886315094 0.00262771165447159 " + "-1.00287836503009 -0.0340303311076584 0.00523658598538616 " + "0.00210057328569037 0.00311664695421511 " + "0.000406267962891555 -0.00550089873884607 " + "5.26202553862825e-05 2.5958707865627e-06 " + "-2.23632198653199e-06 2.02822452334703e-05 " + "-5.24009408417096e-06 2.16913023637956e-05 " + "-2.36002554032361e-05 1.56703932477487e-06 " + "-6.14095576183939e-06 1.8422191786661e-05 " + "2.01615774485329e-07", + "LINE_OFF": "15834", + "LINE_SCALE": "15834", + "LONG_OFF": "147.2588", + "LONG_SCALE": "0.0828", + "SAMP_DEN_COEFF": "1 -0.00520769693945429 0.00549138470185778 " + "-0.00472718007439368 -2.02057365543208e-05 " + "8.39509604990839e-06 -3.43728317291176e-05 " + "-1.94598141718645e-05 2.35660616018416e-05 " + "3.71653924834455e-06 7.08406853351595e-08 " + "-5.72002373463043e-09 8.71363385561163e-09 " + "-4.39082343385483e-09 1.39161295297537e-08 " + "-1.24914964124471e-08 4.6171143198849e-08 " + "1.70067569381897e-08 -2.37449991316111e-08 " + "9.05460084990073e-10", + "SAMP_NUM_COEFF": "0.00121386429106107 1.0047652205584 0.00363040578479236 " + "-0.033702055558379 0.00665808526532754 -0.0036225246184886 " + "-0.000251174639124558 -0.00523232964739848 " + "1.74629130199178e-05 0.000131049715749837 " + "-2.8029900276216e-05 -1.95273546561053e-05 " + "2.91977037724833e-05 2.78736117374265e-07 " + "-2.64005759423042e-05 7.08277764663844e-08 " + "1.12211574945991e-06 1.15330773275834e-06 " + "7.15463416011193e-07 -2.42741993962372e-08", + "SAMP_OFF": "13464", + "SAMP_SCALE": "13464", + } + assert len(got_md) == len(expected_md) + for key in expected_md: + expected_value = expected_md[key] + got_value = got_md[key] + if " " in expected_value: + expected_values = expected_value.split(" ") + got_values = got_value.split(" ") + for i in range(len(expected_values)): + assert float(got_values[i]) == pytest.approx( + float(expected_values[i]), rel=1e-10 + ) + else: + assert float(got_value) == pytest.approx(float(expected_value), rel=1e-10) + + +############################################################################### +# Corrupt all bytes of an existing file + + +def test_libertiff_corrupted(tmp_vsimem): + + filename = str(tmp_vsimem / "out.tif") + data = open("data/gtiff/miniswhite.tif", "rb").read() + gdal.FileFromMemBuffer(filename, data) + f = gdal.VSIFOpenL(filename, "rb+") + try: + for i in range(len(data)): + gdal.VSIFSeekL(f, i, 0) + ori_val = gdal.VSIFReadL(1, 1, f) + + for new_val in (b"\x00", b"\x01", b"\x7F", b"\xFE", b"\xFF"): + gdal.VSIFSeekL(f, i, 0) + gdal.VSIFWriteL(new_val, 1, 1, f) + try: + ds = libertiff_open(filename) + ds.ReadRaster() + except Exception: + pass + + # Restore old value + gdal.VSIFSeekL(f, i, 0) + gdal.VSIFWriteL(ori_val, 1, 1, f) + finally: + gdal.VSIFCloseL(f) + + +def test_libertiff_srs_read_epsg4326_3855_geotiff1_1(): + ds = libertiff_open("data/epsg4326_3855_geotiff1_1.tif") + sr = ds.GetSpatialRef() + assert sr.GetName() == "WGS 84 + EGM2008 height" + assert sr.GetAuthorityCode("COMPD_CS|GEOGCS") == "4326" + assert sr.GetAuthorityCode("COMPD_CS|VERT_CS") == "3855" + + +def test_libertiff_srs_read_epsg4979_geotiff1_1(): + ds = libertiff_open("data/epsg4979_geotiff1_1.tif") + sr = ds.GetSpatialRef() + assert sr.GetAuthorityCode(None) == "4979" + + +def test_libertiff_strip_block_size(): + ds = libertiff_open("data/utmsmall.tif") + assert ds.GetRasterBand(1).GetBlockSize() == [100, 81] + + +def test_libertiff_non_square_pixels(): + ds = libertiff_open("data/gtiff/non_square_pixels.tif") + assert ds.GetGeoTransform() == (2.0, 1.0, 0, 50.0, 0, -0.5) + + +############################################################################### +# Test reading a GeoTIFF file with a geomatrix (ie non zero rotation terms) + + +def test_libertiff_read_geomatrix(): + + ds = libertiff_open("data/geomatrix.tif") + assert ds.GetGeoTransform() == (1841001.75, 1.5, -5.0, 1144003.25, -5.0, -1.5) + + +def test_libertiff_num_threads_saturated(): + + libertiff_open("data/byte.tif", open_options=["NUM_THREADS=10000"]) diff --git a/autotest/gcore/multidim.py b/autotest/gcore/multidim.py index 0cf75f777868..d25b0be84a63 100644 --- a/autotest/gcore/multidim.py +++ b/autotest/gcore/multidim.py @@ -797,6 +797,136 @@ def test_multidim_asclassicsubdataset_band_metadata(): } +@gdaltest.enable_exceptions() +def test_multidim_asclassicsubdataset_band_imagery_metadata(): + + drv = gdal.GetDriverByName("MEM") + mem_ds = drv.CreateMultiDimensional("myds") + rg = mem_ds.GetRootGroup() + + dimOther = rg.CreateDimension("other", None, None, 2) + + other = rg.CreateMDArray( + "other", [dimOther], gdal.ExtendedDataType.Create(gdal.GDT_Float64) + ) + other.Write(array.array("d", [10.5, 20])) + + fwhm = rg.CreateMDArray( + "fwhm", [dimOther], gdal.ExtendedDataType.Create(gdal.GDT_Float64) + ) + fwhm.Write(array.array("d", [3.5, 4.5])) + + string_attr = other.CreateAttribute( + "string_attr", [], gdal.ExtendedDataType.CreateString() + ) + assert string_attr.Write("nm") == gdal.CE_None + + dimX = rg.CreateDimension("X", None, None, 2) + X = rg.CreateMDArray("X", [dimX], gdal.ExtendedDataType.Create(gdal.GDT_Float64)) + X.Write(array.array("d", [10, 20])) + dimX.SetIndexingVariable(X) + dimY = rg.CreateDimension("Y", None, None, 2) + + ar = rg.CreateMDArray( + "ar", [dimOther, dimY, dimX], gdal.ExtendedDataType.Create(gdal.GDT_Float64) + ) + + with pytest.raises( + Exception, + match="Root group should be provided when BAND_IMAGERY_METADATA is set", + ): + ar.AsClassicDataset(2, 1, None, ["BAND_IMAGERY_METADATA={}"]) + + with pytest.raises( + Exception, match="Invalid JSON content for BAND_IMAGERY_METADATA" + ): + ar.AsClassicDataset(2, 1, rg, ["BAND_IMAGERY_METADATA=invalid"]) + + with pytest.raises( + Exception, match="Value of BAND_IMAGERY_METADATA should be an object" + ): + ar.AsClassicDataset(2, 1, rg, ["BAND_IMAGERY_METADATA=false"]) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {}} + with pytest.raises( + Exception, + match=r'BAND_IMAGERY_METADATA\["CENTRAL_WAVELENGTH_UM"\]\["array"\] is missing', + ): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {"array": "/i/do/not/exist"}} + with pytest.raises(Exception, match="Array /i/do/not/exist cannot be found"): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {"array": "/ar"}} + with pytest.raises(Exception, match="Array /ar is not a 1D array"): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {"array": "/X"}} + with pytest.raises( + Exception, + match='Dimension "X" of array "/X" is not a non-X/Y dimension of array "ar"', + ): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {"array": "/other", "unit": "${invalid"}} + with pytest.raises( + Exception, + match=r'Value of BAND_IMAGERY_METADATA\["CENTRAL_WAVELENGTH_UM"\]\["unit"\] = \${invalid is invalid', + ): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {"array": "/other", "unit": "${invalid}"}} + with pytest.raises( + Exception, + match=r'Value of BAND_IMAGERY_METADATA\["CENTRAL_WAVELENGTH_UM"\]\["unit"\] = \${invalid} is invalid: invalid is not an attribute of \/other', + ): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"CENTRAL_WAVELENGTH_UM": {"array": "/other", "unit": "unhandled"}} + with pytest.raises( + Exception, + match=r'Unhandled value for BAND_IMAGERY_METADATA\["CENTRAL_WAVELENGTH_UM"\]\["unit"\] = unhandled', + ): + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = {"ignored": {}} + with gdal.quiet_errors(): + ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + + band_metadata = { + "CENTRAL_WAVELENGTH_UM": {"array": "/other", "unit": "${string_attr}"}, + "FWHM_UM": {"array": "/fwhm"}, + } + ds = ar.AsClassicDataset( + 2, 1, rg, ["BAND_IMAGERY_METADATA=" + json.dumps(band_metadata)] + ) + assert ds.GetRasterBand(1).GetMetadata("IMAGERY") == { + "CENTRAL_WAVELENGTH_UM": "0.0105", + "FWHM_UM": "3.5", + } + assert ds.GetRasterBand(2).GetMetadata("IMAGERY") == { + "CENTRAL_WAVELENGTH_UM": "0.02", + "FWHM_UM": "4.5", + } + + @gdaltest.enable_exceptions() def test_multidim_SubsetDimensionFromSelection(): diff --git a/autotest/gcore/vsicurl.py b/autotest/gcore/vsicurl.py index 2bc0f2f2b4b4..43c7de4bb761 100755 --- a/autotest/gcore/vsicurl.py +++ b/autotest/gcore/vsicurl.py @@ -1609,3 +1609,27 @@ def test_vsicurl_VSICURL_QUERY_STRING(server, filename, query_string): assert statres.size == 3 finally: gdal.SetPathSpecificOption(full_filename, "VSICURL_QUERY_STRING", None) + + +############################################################################### +# Test /vsicurl?header.foo=bar& + + +@gdaltest.enable_exceptions() +def test_vsicurl_header_option(server): + + gdal.VSICurlClearCache() + + handler = webserver.SequentialHandler() + handler.add( + "HEAD", + "/test_vsicurl_header_option.bin", + 200, + {"Content-Length": "3"}, + expected_headers={"foo": "bar", "Accept": "application/json"}, + ) + + with webserver.install_http_handler(handler): + full_filename = f"/vsicurl?header.foo=bar&header.Accept=application%2Fjson&url=http%3A%2F%2Flocalhost%3A{server.port}%2Ftest_vsicurl_header_option.bin" + statres = gdal.VSIStatL(full_filename) + assert statres.size == 3 diff --git a/autotest/gdrivers/vrtprocesseddataset.py b/autotest/gdrivers/vrtprocesseddataset.py index fe076b8b8b08..633f2158a654 100755 --- a/autotest/gdrivers/vrtprocesseddataset.py +++ b/autotest/gdrivers/vrtprocesseddataset.py @@ -123,10 +123,13 @@ def test_vrtprocesseddataset_errors(tmp_vsimem): # Test nominal cases of BandAffineCombination algorithm -def test_vrtprocesseddataset_affine_combination_nominal(tmp_vsimem): +@pytest.mark.parametrize("INTERLEAVE", ["PIXEL", "BAND"]) +def test_vrtprocesseddataset_affine_combination_nominal(tmp_vsimem, INTERLEAVE): src_filename = str(tmp_vsimem / "src.tif") - src_ds = gdal.GetDriverByName("GTiff").Create(src_filename, 2, 1, 3) + src_ds = gdal.GetDriverByName("GTiff").Create( + src_filename, 2, 1, 3, options=["INTERLEAVE=" + INTERLEAVE] + ) src_ds.GetRasterBand(1).WriteArray(np.array([[1, 3]])) src_ds.GetRasterBand(2).WriteArray(np.array([[2, 6]])) src_ds.GetRasterBand(3).WriteArray(np.array([[3, 3]])) @@ -1323,3 +1326,194 @@ def test_vrtprocesseddataset_OutputBands(): match="Invalid value for OutputBands.dataType", ): gdal.Open("data/vrt/processed_OutputBands_USER_PROVIDED_invalid_type.vrt") + + +############################################################################### +# Test VRTProcessedDataset::RasterIO() + + +def test_vrtprocesseddataset_RasterIO(tmp_vsimem): + + src_filename = str(tmp_vsimem / "src.tif") + src_ds = gdal.GetDriverByName("GTiff").Create(src_filename, 2, 3, 4) + src_ds.GetRasterBand(1).WriteArray(np.array([[1, 2], [3, 4], [5, 6]])) + src_ds.GetRasterBand(2).WriteArray(np.array([[7, 8], [9, 10], [11, 12]])) + src_ds.GetRasterBand(3).WriteArray(np.array([[13, 14], [15, 16], [17, 18]])) + src_ds.GetRasterBand(4).WriteArray(np.array([[19, 20], [21, 22], [23, 24]])) + src_ds.BuildOverviews("NEAR", [2]) + src_ds = None + + vrt_content = f""" + + {src_filename} + + + + BandAffineCombination + 0,0,1,0,0 + 0,0,0,1,0 + 0,0,0,0,1 + 0,1,0,0,0 + + + + """ + + ds = gdal.Open(vrt_content) + assert ds.RasterXSize == 2 + assert ds.RasterYSize == 3 + assert ds.RasterCount == 4 + + # Optimized code path with INTERLEAVE=BAND + np.testing.assert_equal( + ds.ReadAsArray(), + np.array( + [ + [[7, 8], [9, 10], [11, 12]], + [[13, 14], [15, 16], [17, 18]], + [[19, 20], [21, 22], [23, 24]], + [[1, 2], [3, 4], [5, 6]], + ] + ), + ) + + # Optimized code path with INTERLEAVE=BAND but buf_type != native type + np.testing.assert_equal( + ds.ReadAsArray(buf_type=gdal.GDT_Int16), + np.array( + [ + [[7, 8], [9, 10], [11, 12]], + [[13, 14], [15, 16], [17, 18]], + [[19, 20], [21, 22], [23, 24]], + [[1, 2], [3, 4], [5, 6]], + ] + ), + ) + + # Optimized code path with INTERLEAVE=BAND + np.testing.assert_equal( + ds.ReadAsArray(1, 2, 1, 1), + np.array([[[12]], [[18]], [[24]], [[6]]]), + ) + + # Optimized code path with INTERLEAVE=PIXEL + np.testing.assert_equal( + ds.ReadAsArray(interleave="PIXEL"), + np.array( + [ + [[7, 13, 19, 1], [8, 14, 20, 2]], + [[9, 15, 21, 3], [10, 16, 22, 4]], + [[11, 17, 23, 5], [12, 18, 24, 6]], + ] + ), + ) + + # Optimized code path with INTERLEAVE=PIXEL but buf_type != native type + np.testing.assert_equal( + ds.ReadAsArray(interleave="PIXEL", buf_type=gdal.GDT_Int16), + np.array( + [ + [[7, 13, 19, 1], [8, 14, 20, 2]], + [[9, 15, 21, 3], [10, 16, 22, 4]], + [[11, 17, 23, 5], [12, 18, 24, 6]], + ] + ), + ) + + # Optimized code path with INTERLEAVE=PIXEL + np.testing.assert_equal( + ds.ReadAsArray(1, 2, 1, 1, interleave="PIXEL"), + np.array([[[12, 18, 24, 6]]]), + ) + + # Not optimized INTERLEAVE=BAND because not enough bands + np.testing.assert_equal( + ds.ReadAsArray(band_list=[1, 2, 3]), + np.array( + [ + [[7, 8], [9, 10], [11, 12]], + [[13, 14], [15, 16], [17, 18]], + [[19, 20], [21, 22], [23, 24]], + ] + ), + ) + + # Not optimized INTERLEAVE=BAND because of out-of-order band list + np.testing.assert_equal( + ds.ReadAsArray(band_list=[4, 1, 2, 3]), + np.array( + [ + [[1, 2], [3, 4], [5, 6]], + [[7, 8], [9, 10], [11, 12]], + [[13, 14], [15, 16], [17, 18]], + [[19, 20], [21, 22], [23, 24]], + ] + ), + ) + + # Not optimized INTERLEAVE=PIXEL because of out-of-order band list + np.testing.assert_equal( + ds.ReadAsArray(interleave="PIXEL", band_list=[4, 1, 2, 3]), + np.array( + [ + [[1, 7, 13, 19], [2, 8, 14, 20]], + [[3, 9, 15, 21], [4, 10, 16, 22]], + [[5, 11, 17, 23], [6, 12, 18, 24]], + ] + ), + ) + + # Optimized code path with overviews + assert ds.GetRasterBand(1).GetOverview(0).XSize == 1 + assert ds.GetRasterBand(1).GetOverview(0).YSize == 2 + np.testing.assert_equal( + ds.ReadAsArray(buf_xsize=1, buf_ysize=2), + np.array([[[7], [11]], [[13], [17]], [[19], [23]], [[1], [5]]]), + ) + + # Non-optimized code path with overviews + np.testing.assert_equal( + ds.ReadAsArray(buf_xsize=1, buf_ysize=1), + np.array([[[11]], [[17]], [[23]], [[5]]]), + ) + + # Test buffer splitting + with gdal.config_option("VRT_PROCESSED_DATASET_ALLOWED_RAM_USAGE", "96"): + ds = gdal.Open(vrt_content) + + # Optimized code path with INTERLEAVE=BAND + np.testing.assert_equal( + ds.ReadAsArray(), + np.array( + [ + [[7, 8], [9, 10], [11, 12]], + [[13, 14], [15, 16], [17, 18]], + [[19, 20], [21, 22], [23, 24]], + [[1, 2], [3, 4], [5, 6]], + ] + ), + ) + + # I/O error + gdal.GetDriverByName("GTiff").Create( + src_filename, 1024, 1024, 4, options=["TILED=YES"] + ) + f = gdal.VSIFOpenL(src_filename, "rb+") + gdal.VSIFTruncateL(f, 4096) + gdal.VSIFCloseL(f) + + ds = gdal.Open(vrt_content) + + # Error in INTERLEAVE=BAND optimized code path + with pytest.raises(Exception): + ds.ReadAsArray() + + # Error in INTERLEAVE=PIXEL optimized code path + with pytest.raises(Exception): + ds.ReadAsArray(interleave="PIXEL") + + with gdal.config_option("VRT_PROCESSED_DATASET_ALLOWED_RAM_USAGE", "96"): + ds = gdal.Open(vrt_content) + assert ds.GetRasterBand(1).GetBlockSize() == [1, 1] + with pytest.raises(Exception): + ds.ReadAsArray() diff --git a/autotest/lsan_suppressions.txt b/autotest/lsan_suppressions.txt index 9c3329c37dbd..e4547caf428f 100644 --- a/autotest/lsan_suppressions.txt +++ b/autotest/lsan_suppressions.txt @@ -12,3 +12,5 @@ leak:PyDataMem_NEW leak:PyArray_NewFromDescr_int leak:/usr/bin/python3.8 leak:/usr/bin/python3.10 +leak:/usr/bin/python3.12 +leak:/lib/x86_64-linux-gnu/libduckdb.so diff --git a/autotest/ogr/data/geojson/stac_item.json b/autotest/ogr/data/geojson/stac_item.json new file mode 100644 index 000000000000..05821ff9c8c7 --- /dev/null +++ b/autotest/ogr/data/geojson/stac_item.json @@ -0,0 +1,109 @@ +{ + "type": "Feature", + "stac_version": "1.0.0", + "stac_extensions": [ + "https://stac-extensions.github.io/projection/v1.0.0/schema.json", + ], + "id": "my_id", + "description": "Landsat Collection 2 Level-2 Surface Reflectance Product", + "bbox": [ + -155.80483200278817, + 17.736060531368373, + -153.68026753986817, + 19.82593799656933 + ], + "geometry": { + "type": "Polygon", + "coordinates": [ + [ + [ + -155.41188228421524, + 19.82593799656933 + ], + [ + -155.80483200278817, + 18.09429339392015 + ], + [ + -154.0866457786547, + 17.736060531368373 + ], + [ + -153.68026753986817, + 19.470851166420736 + ], + [ + -155.41188228421524, + 19.82593799656933 + ] + ] + ] + }, + "properties": { + "datetime": "2024-12-16T20:42:39.252121Z", + "eo:cloud_cover": 58.93, + "view:sun_azimuth": 150.92260609, + "view:sun_elevation": 42.2078599, + "platform": "LANDSAT_9", + "instruments": [ + "OLI", + "TIRS" + ], + "view:off_nadir": 0, + "landsat:cloud_cover_land": 43.14, + "landsat:wrs_type": "2", + "landsat:wrs_path": "062", + "landsat:wrs_row": "047", + "landsat:scene_id": "LC90620472024351LGN00", + "landsat:collection_category": "A1", + "landsat:collection_number": "02", + "landsat:correction": "L2SP", + "accuracy:geometric_x_bias": 0, + "accuracy:geometric_y_bias": 0, + "accuracy:geometric_x_stddev": 5.048, + "accuracy:geometric_y_stddev": 4.77, + "accuracy:geometric_rmse": 6.946, + "proj:epsg": null, + "proj:shape": [ + 7701, + 7571 + ], + "proj:transform": [ + 30, + 0, + 124185, + 0, + -30, + 1862415 + ], + "proj:wkt2": "PROJCS[\"AEA WGS84\",GEOGCS[\"WGS 84\",DATUM[\"WGS_1984\",SPHEROID[\"WGS 84\",6378137,298.257223563,AUTHORITY[\"EPSG\",\"7030\"]],AUTHORITY[\"EPSG\",\"6326\"]],PRIMEM[\"Greenwich\",0],UNIT[\"degree\",0.0174532925199433,AUTHORITY[\"EPSG\",\"9122\"]],AUTHORITY[\"EPSG\",\"4326\"]],PROJECTION[\"Albers_Conic_Equal_Area\"],PARAMETER[\"latitude_of_center\",3],PARAMETER[\"longitude_of_center\",-157],PARAMETER[\"standard_parallel_1\",8],PARAMETER[\"standard_parallel_2\",18],PARAMETER[\"false_easting\",0],PARAMETER[\"false_northing\",0],UNIT[\"metre\",1,AUTHORITY[\"EPSG\",\"9001\"]],AXIS[\"Easting\",EAST],AXIS[\"Northing\",NORTH]]", + "card4l:specification": "SR", + "card4l:specification_version": "5.0", + "created": "2024-12-17T13:13:26.984Z", + "updated": "2024-12-18T04:59:46.657Z" + }, + "assets": { + "thumbnail": { + "title": "Thumbnail image", + "type": "image/jpeg", + "roles": [ + "thumbnail" + ], + "href": "https://exmaple.com/thumb_small.jpeg", + "alternate": { + "s3": { + "storage:platform": "AWS", + "storage:requester_pays": true, + "href": "s3://example/thumb_small.jpeg" + } + } + } + }, + "links": [ + { + "rel": "self", + "href": "https://example.com/items/my_id" + } + ], + "collection": "landsat-c2l2alb-sr" +} diff --git a/autotest/ogr/ogr_geojson.py b/autotest/ogr/ogr_geojson.py index 61e25654c9a9..ee83ba27c291 100755 --- a/autotest/ogr/ogr_geojson.py +++ b/autotest/ogr/ogr_geojson.py @@ -5784,3 +5784,39 @@ def test_ogr_geojson_schema_override( assert ( gdal.GetLastErrorMsg().find(expected_warning) != -1 ), f"Warning {expected_warning} not found, got {gdal.GetLastErrorMsg()} instead" + + +############################################################################### +# Test FOREIGN_MEMBERS open option + + +@pytest.mark.parametrize( + "foreign_members_option", [None, "AUTO", "ALL", "NONE", "STAC"] +) +def test_ogr_geojson_foreign_members(foreign_members_option): + + open_options = {} + if foreign_members_option: + open_options["FOREIGN_MEMBERS"] = foreign_members_option + ds = gdal.OpenEx( + "data/geojson/stac_item.json", gdal.OF_VECTOR, open_options=open_options + ) + lyr = ds.GetLayer(0) + f = lyr.GetNextFeature() + if foreign_members_option is None or foreign_members_option in ("AUTO", "STAC"): + assert lyr.GetLayerDefn().GetFieldCount() == 39 + assert f["stac_version"] == "1.0.0" + assert f["assets.thumbnail.title"] == "Thumbnail image" + assert json.loads(f["assets.thumbnail.alternate"]) == { + "s3": { + "storage:platform": "AWS", + "storage:requester_pays": True, + "href": "s3://example/thumb_small.jpeg", + } + } + elif foreign_members_option == "ALL": + assert lyr.GetLayerDefn().GetFieldCount() == 35 + assert f["stac_version"] == "1.0.0" + assert f["assets"] != "" + else: + assert lyr.GetLayerDefn().GetFieldCount() == 29 diff --git a/autotest/ogr/ogr_miramon_vector.py b/autotest/ogr/ogr_miramon_vector.py index 3cfebb9b3ba6..ecf4a647c9bb 100644 --- a/autotest/ogr/ogr_miramon_vector.py +++ b/autotest/ogr/ogr_miramon_vector.py @@ -1437,9 +1437,14 @@ def test_ogr_miramon_create_field_after_feature(tmp_path): def test_ogr_miramon_json_import_not_failing(tmp_vsimem): out_filename = str(tmp_vsimem / "out/json_layer_to_mm.pol") + src_ds = gdal.OpenEx( + "data/miramon_inputs/LT05_L2SP_038037_20120505_20200820_02_T1_ST_stac_minimal.json", + gdal.OF_VECTOR, + open_options=["FOREIGN_MEMBERS=NONE"], + ) gdal.VectorTranslate( out_filename, - "data/miramon_inputs/LT05_L2SP_038037_20120505_20200820_02_T1_ST_stac_minimal.json", + src_ds, format="MiraMonVector", ) diff --git a/autotest/pymod/gdaltest.py b/autotest/pymod/gdaltest.py index 77da5d307872..a17d6bce3ada 100755 --- a/autotest/pymod/gdaltest.py +++ b/autotest/pymod/gdaltest.py @@ -218,6 +218,14 @@ def testOpen( drv_name.lower() == "biggif" and self.drivername.lower() == "gif" ) + or ( + drv_name.lower() == "gtiff" + and self.drivername.lower() == "libertiff" + ) + or ( + drv_name.lower() == "libertiff" + and self.drivername.lower() == "gtiff" + ) ): drivers += [drv_name] other_ds = None diff --git a/autotest/utilities/test_gdal.py b/autotest/utilities/test_gdal.py index 420b6e9a0c98..ef1f50efaf18 100755 --- a/autotest/utilities/test_gdal.py +++ b/autotest/utilities/test_gdal.py @@ -17,6 +17,8 @@ import pytest import test_cli_utilities +from osgeo import gdal + pytestmark = pytest.mark.skipif( test_cli_utilities.get_gdal_path() is None, reason="gdal binary not available" ) @@ -84,3 +86,311 @@ def test_gdal_failure_during_finalize(gdal_path): f"{gdal_path} raster convert ../gcore/data/byte.tif /vsimem/out.tif||maxlength=100" ) assert "ret code = 1" in err + + +def test_gdal_completion(gdal_path): + + out = gdaltest.runexternal(f"{gdal_path} completion gdal -").split(" ") + assert "--version" in out + + out = gdaltest.runexternal(f"{gdal_path} completion gdal").split(" ") + assert "convert" in out + assert "info" in out + assert "raster" in out + assert "vector" in out + + out = gdaltest.runexternal(f"{gdal_path} completion gdal raster").split(" ") + assert "convert" in out + assert "edit" in out + assert "info" in out + assert "reproject" in out + assert "pipeline" in out + + out = gdaltest.runexternal(f"{gdal_path} completion gdal raster info -").split(" ") + assert "-f" in out + assert "--of" in out + assert "--output-format" in out + + out = gdaltest.runexternal(f"{gdal_path} completion gdal raster info --of").split( + " " + ) + assert "json" in out + assert "text" in out + + out = gdaltest.runexternal(f"{gdal_path} completion gdal raster info --of=").split( + " " + ) + assert "json" in out + assert "text" in out + + out = gdaltest.runexternal(f"{gdal_path} completion gdal raster info --of=t").split( + " " + ) + assert "json" in out + assert "text" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --of" + ).split(" ") + if gdal.GetDriverByName("GTiff"): + assert "GTiff" in out + if gdal.GetDriverByName("HFA"): + assert "HFA" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --input" + ).split(" ") + assert "data/" in out or "data\\" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --input data/" + ).split(" ") + assert "data/whiteblackred.tif" in out or "data\\whiteblackred.tif" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert /vsizip/../gcore/data/byte.tif.zip/" + ).split(" ") + assert out == ["/vsizip/../gcore/data/byte.tif.zip/byte.tif"] + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --resolution" + ) + assert ( + out + == "** description:\\ Target\\ resolution\\ (in\\ destination\\ CRS\\ units)" + ) + + +def test_gdal_completion_co(gdal_path): + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co" + ).split(" ") + assert "COMPRESS=" in out + assert "RPCTXT=" in out + assert "TILING_SCHEME=" not in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co=" + ).split(" ") + assert "COMPRESS=" in out + assert "RPCTXT=" in out + assert "TILING_SCHEME=" not in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co COMPRESS=" + ).split(" ") + assert "NONE" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co COMPRESS=NO" + ).split(" ") + assert "NONE" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co=COMPRESS=" + ).split(" ") + assert "NONE" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co=COMPRESS=NO" + ).split(" ") + assert "NONE" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co TILED=" + ).split(" ") + assert out == ["NO", "YES"] + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co ZLEVEL=" + ).split(" ") + assert "1" in out + assert "9" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --of COG --co=" + ).split(" ") + assert "COMPRESS=" in out + assert "RPCTXT=" not in out + assert "TILING_SCHEME=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --of=COG --creation-option ZOOM_LEVEL=" + ) + assert ( + out + == "## type:\\ int,\\ description:\\ Target\\ zoom\\ level.\\ Only\\ used\\ for\\ TILING_SCHEME\\ !=\\ CUSTOM" + ) + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --of COG --co BLOCKSIZE=" + ) + assert out == "## validity\\ range:\\ >=\\ 128" + + if "JPEG_QUALITY" in gdal.GetDriverByName("GTiff").GetMetadataItem( + "DMD_CREATIONOPTIONLIST" + ): + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert in.tif out.tif --co JPEG_QUALITY=" + ) + assert out == "## validity\\ range:\\ [1,100]" + + if gdal.GetDriverByName("GPKG"): + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --of GPKG --co=" + ).split(" ") + assert "APPEND_SUBDATASET=" in out + assert "VERSION=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal vector convert --of GPKG --co=" + ).split(" ") + assert "APPEND_SUBDATASET=" not in out + assert "VERSION=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --of GPKG --co BLOCKSIZE=" + ) + assert out == "## validity\\ range:\\ <=\\ 4096" + + +def test_gdal_completion_lco(gdal_path): + + if gdal.GetDriverByName("GPKG"): + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal vector convert --of GPKG --lco" + ).split(" ") + assert "FID=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal vector convert in.shp out.gpkg --layer-creation-option=" + ).split(" ") + assert "FID=" in out + + +def test_gdal_completion_oo(gdal_path): + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster info foo.tif --oo" + ).split(" ") + assert "COLOR_TABLE_MULTIPLIER=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster info --if GTiff --open-option" + ).split(" ") + assert "COLOR_TABLE_MULTIPLIER=" in out + + +def test_gdal_completion_dst_crs(gdal_path): + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --dst-crs" + ).split(" ") + assert "EPSG:" in out + assert "ESRI:" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --dst-crs EPSG:" + ) + assert "4326\\ --\\ WGS\\ 84" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --dst-crs=EPSG:" + ) + assert "4326\\ --\\ WGS\\ 84" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --dst-crs EPSG:43" + ) + assert "4326\\ --\\ WGS\\ 84" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --dst-crs=EPSG:43" + ) + assert "4326\\ --\\ WGS\\ 84" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster reproject --dst-crs EPSG:4326" + ) + assert out == "4326" + + +def test_gdal_completion_config(gdal_path): + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --config" + ).split(" ") + assert "CPL_DEBUG=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --config=" + ).split(" ") + assert "CPL_DEBUG=" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --config FOO=" + ).split(" ") + assert out == [""] + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal raster convert --config=FOO=" + ).split(" ") + assert out == [""] + + +@pytest.mark.parametrize("subcommand", ["raster", "vector"]) +def test_gdal_completion_pipeline(gdal_path, subcommand): + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline" + ).split(" ") + assert out == ["read"] + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline re" + ).split(" ") + assert out == ["read"] + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read" + ).split(" ") + assert out == [""] + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read -" + ).split(" ") + assert "--input" in out + assert "--open-option" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read !" + ).split(" ") + assert "reproject" in out + assert "write" in out + assert "read" not in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read ! re" + ).split(" ") + assert "reproject" in out + assert "write" in out + assert "read" not in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read foo ! write -" + ).split(" ") + assert "--output" in out + assert "--co" in out + + if subcommand == "raster": + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read foo ! foo ! reproject -" + ).split(" ") + assert "--resampling" in out + + out = gdaltest.runexternal( + f"{gdal_path} completion gdal {subcommand} pipeline read foo ! reproject --resampling" + ).split(" ") + assert "near" in out diff --git a/autotest/utilities/test_gdalalg_raster_buildvrt.py b/autotest/utilities/test_gdalalg_raster_buildvrt.py new file mode 100755 index 000000000000..5a944567ffa7 --- /dev/null +++ b/autotest/utilities/test_gdalalg_raster_buildvrt.py @@ -0,0 +1,363 @@ +#!/usr/bin/env pytest +# -*- coding: utf-8 -*- +############################################################################### +# Project: GDAL/OGR Test Suite +# Purpose: 'gdal raster buildvrt' testing +# Author: Even Rouault +# +############################################################################### +# Copyright (c) 2024, Even Rouault +# +# SPDX-License-Identifier: MIT +############################################################################### + +import pytest + +from osgeo import gdal + + +def get_buildvrt_alg(): + reg = gdal.GetGlobalAlgorithmRegistry() + raster = reg.InstantiateAlg("raster") + return raster.InstantiateSubAlgorithm("buildvrt") + + +def test_gdalalg_raster_buildvrt_from_dataset_handle(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set( + [ + gdal.Translate( + "", "../gcore/data/byte.tif", options="-f MEM -srcwin 0 0 10 20" + ), + gdal.Translate( + "", "../gcore/data/byte.tif", options="-f MEM -srcwin 10 0 10 20" + ), + ] + ) + alg.GetArg("output").Set("") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 20 + assert ds.RasterYSize == 20 + assert ds.GetGeoTransform() == pytest.approx( + (440720.0, 60.0, 0.0, 3751320.0, 0.0, -60.0) + ) + assert ds.GetRasterBand(1).Checksum() == 4672 + + +def test_gdalalg_raster_buildvrt_from_dataset_name(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["../gcore/data/byte.tif"]) + alg.GetArg("output").Set("") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 20 + assert ds.RasterYSize == 20 + assert ds.GetGeoTransform() == pytest.approx( + (440720.0, 60.0, 0.0, 3751320.0, 0.0, -60.0) + ) + assert ds.GetRasterBand(1).Checksum() == 4672 + + +def test_gdalalg_raster_buildvrt_overwrite(tmp_vsimem): + + out_filename = str(tmp_vsimem / "out.tif") + + alg = get_buildvrt_alg() + assert alg.ParseRunAndFinalize(["../gcore/data/utmsmall.tif", out_filename]) + + with gdal.Open(out_filename) as ds: + assert ds.GetRasterBand(1).Checksum() == 50054 + + alg = get_buildvrt_alg() + with pytest.raises( + Exception, match="already exists. Specify the --overwrite option" + ): + alg.ParseRunAndFinalize(["../gcore/data/byte.tif", out_filename]) + + alg = get_buildvrt_alg() + assert alg.ParseRunAndFinalize( + ["--overwrite", "../gcore/data/byte.tif", out_filename] + ) + + with gdal.Open(out_filename) as ds: + assert ds.GetRasterBand(1).Checksum() == 4672 + + +def test_gdalalg_raster_buildvrt_separate(tmp_vsimem): + + tmp_filename = str(tmp_vsimem / "tmp.tif") + gdal.Translate(tmp_filename, "../gcore/data/byte.tif", options="-scale 0 255 255 0") + + alg = get_buildvrt_alg() + assert alg.ParseCommandLineArguments( + [ + "--separate", + "../gcore/data/byte.tif", + tmp_filename, + "", + ] + ) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterCount == 2 + assert ds.RasterXSize == 20 + assert ds.RasterYSize == 20 + assert ds.GetGeoTransform() == pytest.approx( + (440720.0, 60.0, 0.0, 3751320.0, 0.0, -60.0) + ) + assert ds.GetRasterBand(1).Checksum() == 4672 + assert ds.GetRasterBand(2).Checksum() == 4563 + + +def test_gdalalg_raster_buildvrt_bbox(): + + alg = get_buildvrt_alg() + assert alg.ParseCommandLineArguments( + [ + "--bbox=440780.0,3750180.0,441860.0,3751260.0", + "../gcore/data/byte.tif", + "", + ] + ) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 18 + assert ds.RasterYSize == 18 + assert ds.GetGeoTransform() == pytest.approx( + (440780.0, 60.0, 0.0, 3751260.0, 0.0, -60.0) + ) + assert ds.GetRasterBand(1).Checksum() == 3695 + + +def test_gdalalg_raster_buildvrt_resolution_average(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src2_ds = gdal.GetDriverByName("MEM").Create("", 2, 2) + src2_ds.SetGeoTransform([3, 0.5, 0, 49, 0, -0.5]) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds, src2_ds]) + alg.GetArg("output").Set("") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 3 + assert ds.RasterYSize == 1 + assert ds.GetGeoTransform() == pytest.approx((2.0, 0.75, 0.0, 49.0, 0.0, -0.75)) + + +def test_gdalalg_raster_buildvrt_resolution_highest(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src2_ds = gdal.GetDriverByName("MEM").Create("", 2, 2) + src2_ds.SetGeoTransform([3, 0.5, 0, 49, 0, -0.5]) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds, src2_ds]) + alg.GetArg("output").Set("") + alg.GetArg("resolution").Set("highest") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 4 + assert ds.RasterYSize == 2 + assert ds.GetGeoTransform() == pytest.approx((2.0, 0.5, 0.0, 49.0, 0.0, -0.5)) + + +def test_gdalalg_raster_buildvrt_resolution_lowest(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src2_ds = gdal.GetDriverByName("MEM").Create("", 2, 2) + src2_ds.SetGeoTransform([3, 0.5, 0, 49, 0, -0.5]) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds, src2_ds]) + alg.GetArg("output").Set("") + alg.GetArg("resolution").Set("lowest") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 2 + assert ds.RasterYSize == 1 + assert ds.GetGeoTransform() == pytest.approx((2.0, 1.0, 0.0, 49.0, 0.0, -1.0)) + + +def test_gdalalg_raster_buildvrt_resolution_custom(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src2_ds = gdal.GetDriverByName("MEM").Create("", 2, 2) + src2_ds.SetGeoTransform([3, 0.5, 0, 49, 0, -0.5]) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds, src2_ds]) + alg.GetArg("output").Set("") + alg.GetArg("resolution").Set("0.5,1") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 4 + assert ds.RasterYSize == 1 + assert ds.GetGeoTransform() == pytest.approx((2.0, 0.5, 0.0, 49.0, 0.0, -1.0)) + + +def test_gdalalg_raster_buildvrt_target_aligned_pixels(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src2_ds = gdal.GetDriverByName("MEM").Create("", 2, 2) + src2_ds.SetGeoTransform([3, 0.5, 0, 49, 0, -0.5]) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds, src2_ds]) + alg.GetArg("output").Set("") + alg.GetArg("resolution").Set("0.3,0.6") + alg.GetArg("target-aligned-pixels").Set(True) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterXSize == 8 + assert ds.RasterYSize == 2 + assert ds.GetGeoTransform() == pytest.approx((1.8, 0.3, 0.0, 49.2, 0.0, -0.6)) + + +def test_gdalalg_raster_buildvrt_resolution_invalid(): + + alg = get_buildvrt_alg() + with pytest.raises( + Exception, + match="resolution: two comma separated positive values should be provided, or 'average', 'highest' or 'lowest'", + ): + alg.GetArg("resolution").Set("invalid") + + with pytest.raises( + Exception, + match="resolution: two comma separated positive values should be provided, or 'average', 'highest' or 'lowest'", + ): + alg.GetArg("resolution").Set("0.5") + + with pytest.raises( + Exception, + match="resolution: two comma separated positive values should be provided, or 'average', 'highest' or 'lowest'", + ): + alg.GetArg("resolution").Set("-0.5,-0.5") + + +def test_gdalalg_raster_buildvrt_srcnodata_vrtnodata(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src1_ds.GetRasterBand(1).Fill(1) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds]) + alg.GetArg("output").Set("") + alg.GetArg("srcnodata").Set([1]) + alg.GetArg("vrtnodata").Set([2]) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.GetRasterBand(1).Checksum() == 2 + assert ds.GetRasterBand(1).GetNoDataValue() == 2 + + +def test_gdalalg_raster_buildvrt_hidenodata(): + + src1_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + src1_ds.SetGeoTransform([2, 1, 0, 49, 0, -1]) + src1_ds.GetRasterBand(1).Fill(1) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([src1_ds]) + alg.GetArg("output").Set("") + alg.GetArg("srcnodata").Set([1]) + alg.GetArg("vrtnodata").Set([2]) + alg.GetArg("hidenodata").Set(True) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.GetRasterBand(1).Checksum() == 2 + assert ds.GetRasterBand(1).GetNoDataValue() is None + + +def test_gdalalg_raster_buildvrt_addalpha(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["../gcore/data/byte.tif"]) + alg.GetArg("output").Set("") + alg.GetArg("addalpha").Set(True) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterCount == 2 + assert ds.GetRasterBand(1).Checksum() == 4672 + assert ds.GetRasterBand(2).GetColorInterpretation() == gdal.GCI_AlphaBand + assert ds.GetRasterBand(2).Checksum() == 4873 + + +def test_gdalalg_raster_buildvrt_band(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["../gcore/data/rgbsmall.tif"]) + alg.GetArg("output").Set("") + alg.GetArg("band").Set([3, 2]) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterCount == 2 + assert ds.GetRasterBand(1).Checksum() == 21349 + assert ds.GetRasterBand(2).Checksum() == 21053 + + +def test_gdalalg_raster_buildvrt_glob(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["../gcore/data/rgbsm?ll.tif"]) + alg.GetArg("output").Set("") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.RasterCount == 3 + + +def test_gdalalg_raster_buildvrt_at_filename(tmp_vsimem): + + input_file_list = str(tmp_vsimem / "tmp.txt") + gdal.FileFromMemBuffer(input_file_list, "../gcore/data/byte.tif") + + alg = get_buildvrt_alg() + alg.GetArg("input").Set([f"@{input_file_list}"]) + alg.GetArg("output").Set("") + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.GetRasterBand(1).Checksum() == 4672 + + +def test_gdalalg_raster_buildvrt_at_filename_error(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["@i_do_not_exist"]) + alg.GetArg("output").Set("") + with pytest.raises(Exception, match="buildvrt: Cannot open i_do_not_exist"): + alg.Run() + + +def test_gdalalg_raster_buildvrt_output_ds_alread_set(): + + out_ds = gdal.GetDriverByName("MEM").Create("", 1, 1) + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["../gcore/data/byte.tif"]) + alg.GetArg("output").Set(out_ds) + with pytest.raises( + Exception, + match="buildvrt: gdal raster buildvrt does not support outputting to an already opened output dataset", + ): + alg.Run() + + +def test_gdalalg_raster_buildvrt_co(): + + alg = get_buildvrt_alg() + alg.GetArg("input").Set(["../gcore/data/byte.tif"]) + alg.GetArg("output").Set("") + alg.GetArg("creation-option").Set(["BLOCKXSIZE=10", "BLOCKYSIZE=15"]) + assert alg.Run() + ds = alg.GetArg("output").Get().GetDataset() + assert ds.GetRasterBand(1).GetBlockSize() == [10, 15] diff --git a/autotest/utilities/test_gdalalg_raster_edit.py b/autotest/utilities/test_gdalalg_raster_edit.py index a07ace225d29..d85556a8f36a 100755 --- a/autotest/utilities/test_gdalalg_raster_edit.py +++ b/autotest/utilities/test_gdalalg_raster_edit.py @@ -28,7 +28,7 @@ def test_gdalalg_raster_edit_read_only(tmp_vsimem): gdal.FileFromMemBuffer(tmp_filename, open("../gcore/data/byte.tif", "rb").read()) pipeline = get_edit_alg() - pipeline.GetArg("dataset").Get().SetDataset(gdal.OpenEx(tmp_filename)) + pipeline.GetArg("dataset").Set(gdal.OpenEx(tmp_filename)) with pytest.raises( Exception, match="edit: Dataset should be opened in update mode" ): diff --git a/doc/source/api/cpl.rst b/doc/source/api/cpl.rst index 0724303fcfa8..3feeee965481 100644 --- a/doc/source/api/cpl.rst +++ b/doc/source/api/cpl.rst @@ -96,3 +96,11 @@ cpl_vsi.h .. seealso:: :ref:`cpl_cpp_api`. + + +.. below is an allow-list for spelling checker. + +.. spelling:word-list:: + subdir + subsubdir + diff --git a/doc/source/api/raster_c_api.rst b/doc/source/api/raster_c_api.rst index feb6477d9d44..8c2dc9b73209 100644 --- a/doc/source/api/raster_c_api.rst +++ b/doc/source/api/raster_c_api.rst @@ -11,3 +11,12 @@ gdal.h: Raster C API .. doxygenfile:: gdal.h :project: api + + +.. below is an allow-list for spelling checker. + +.. spelling:word-list:: + nSrcHeight + nSrcWidth + pSrc + pDst diff --git a/doc/source/conf.py b/doc/source/conf.py index 15d09859b36c..625663f20718 100644 --- a/doc/source/conf.py +++ b/doc/source/conf.py @@ -215,6 +215,13 @@ [author_evenr], 1, ), + ( + "programs/gdal_raster_buildvrt", + "gdal-raster-buildvrt", + "Build a virtual dataset (VRT)", + [author_evenr], + 1, + ), ( "programs/gdal_raster_info", "gdal-raster-info", diff --git a/doc/source/drivers/raster/gtiff.rst b/doc/source/drivers/raster/gtiff.rst index f6d4f722c42a..5f685fec05e6 100644 --- a/doc/source/drivers/raster/gtiff.rst +++ b/doc/source/drivers/raster/gtiff.rst @@ -25,6 +25,9 @@ file, such as YCbCr color model files, are automatically translated into RGBA (red, green, blue, alpha) form, and treated as four eight bit bands. +For an alternative which offers thread-safe read-only capabilities, consult +:ref:`raster.libertiff`. + Driver capabilities ------------------- diff --git a/doc/source/drivers/raster/index.rst b/doc/source/drivers/raster/index.rst index 0e511f99951f..7231e3208a3e 100644 --- a/doc/source/drivers/raster/index.rst +++ b/doc/source/drivers/raster/index.rst @@ -109,6 +109,7 @@ Raster drivers l1b lcp leveller + libertiff loslas map marfa diff --git a/doc/source/drivers/raster/libertiff.rst b/doc/source/drivers/raster/libertiff.rst new file mode 100644 index 000000000000..b13ab47c4a50 --- /dev/null +++ b/doc/source/drivers/raster/libertiff.rst @@ -0,0 +1,50 @@ +.. _raster.libertiff: + +================================================================================ +LIBERTIFF -- GeoTIFF File Format +================================================================================ + +.. versionadded:: 3.11 + +.. shortname:: LIBERTIFF + +.. built_in_by_default:: + +This driver is a natively thread-safe alternative to the default +:ref:`raster.gtiff` driver. Note that the driver is read-only. + +The driver is registered after the GTiff one. Consequently one must explicitly +specify ``LIBERTIFF`` in the allowed drivers of the :cpp:func:`GDALOpenEx`, or +with the ``-if`` option of command line utilities, to use it. + +The driver supports the following compression methods: LZW, Deflate, PackBits, +LZMA, ZSTD, LERC, JPEG, WEBP and JPEGXL. +The driver supports only BitsPerSample values of 1, 8, 16, 32, 64. + +The driver does *not* read any side-car file: ``.aux.xml``, ``.ovr``, ``.msk``, +``.imd``, etc. + +The driver mostly by-passes the GDAL raster block cache, and only caches (per-thread) +the last tile or strip it has read. Read patterns must be adapted accordingly, +to avoid repeated data acquisition from storage and decompression. + +Driver capabilities +------------------- + +.. supports_georeferencing:: + +.. supports_virtualio:: + +Open options +------------ + +|about-open-options| +This driver supports the following open options: + +.. oo:: NUM_THREADS + :choices: , ALL_CPUS + + This option also enables multi-threaded decoding + when RasterIO() requests intersect several tiles/strips. + The :config:`GDAL_NUM_THREADS` configuration option can also + be used as an alternative to setting the open option. diff --git a/doc/source/drivers/vector/geojson.rst b/doc/source/drivers/vector/geojson.rst index 4bc519abb7d6..dab99b5ec820 100644 --- a/doc/source/drivers/vector/geojson.rst +++ b/doc/source/drivers/vector/geojson.rst @@ -239,6 +239,27 @@ This driver supports the following open options: Can also be set with the :config:`OGR_GEOJSON_DATE_AS_STRING` configuration option. +- .. oo:: FOREIGN_MEMBERS + :choices: AUTO, ALL, NONE, STAC + :default: AUTO + :since: 3.11.0 + + Whether and how foreign members at the feature level should be processed + as OGR fields: + + - ``AUTO`` mode behaves like ``STAC`` mode if a ``stac_version`` member is found at + the Feature level, otherwise it behaves as ``NONE`` mode. + + - In ``ALL`` mode, all foreign members at the feature level are added. + Whether to recursively explore nested objects and produce flatten OGR attributes + or not is decided by the ``FLATTEN_NESTED_ATTRIBUTES`` open option. + + - In ``NONE`` mode, no foreign members at the feature level are added. + + - ``STAC`` mode (Spatio-Temporal Asset Catalog) behaves the same as ``ALL``, + except content under the ``assets`` member is by default flattened + as ``assets.{asset_name}.{asset_property}`` fields. + - .. oo:: OGR_SCHEMA :choices: | :since: 3.11.0 diff --git a/doc/source/faq.rst b/doc/source/faq.rst index b079349bd59d..77697ae853bd 100644 --- a/doc/source/faq.rst +++ b/doc/source/faq.rst @@ -92,6 +92,15 @@ How do I add support for a new format? To some extent this is now covered by the :ref:`raster_driver_tut` and :ref:`vector_driver_tut` +What about file formats for 3D models? +++++++++++++++++++++++++++++++++++++++ + +While some vector drivers in GDAL support 3D geometries (e.g. reading CityGML or +AIXM through the GML driver, or 3D geometries in GeoPackage), 3-dimensional scenes or +models formats (such as glTF, OBJ, 3DS, etc.) are out-of-scope for GDAL. +Users may consider a library such as `ASSIMP (Open-Asset-Importer-Library) `__ +for such formats. + Is GDAL thread-safe ? +++++++++++++++++++++ @@ -108,3 +117,9 @@ How do I cite GDAL ? See `CITATION`_ .. _`CITATION`: https://github.com/OSGeo/gdal/blob/master/CITATION + + +.. below is an allow-list for spelling checker. + +.. spelling:word-list:: + glTF diff --git a/doc/source/programs/gdal_raster.rst b/doc/source/programs/gdal_raster.rst index 118bf2074caf..67cd7bd12fe1 100644 --- a/doc/source/programs/gdal_raster.rst +++ b/doc/source/programs/gdal_raster.rst @@ -19,6 +19,7 @@ Synopsis Usage: gdal raster where is one of: + - buildvrt: Build a virtual dataset (VRT). - convert: Convert a raster dataset. - info: Return information on a raster dataset. - pipeline: Process a raster dataset. @@ -27,6 +28,7 @@ Synopsis Available sub-commands ---------------------- +- :ref:`gdal_raster_buildvrt_subcommand` - :ref:`gdal_raster_info_subcommand` - :ref:`gdal_raster_convert_subcommand` - :ref:`gdal_raster_pipeline_subcommand` diff --git a/doc/source/programs/gdal_raster_buildvrt.rst b/doc/source/programs/gdal_raster_buildvrt.rst new file mode 100644 index 000000000000..ce29983e665f --- /dev/null +++ b/doc/source/programs/gdal_raster_buildvrt.rst @@ -0,0 +1,181 @@ +.. _gdal_raster_buildvrt_subcommand: + +================================================================================ +"gdal raster buildvrt" sub-command +================================================================================ + +.. versionadded:: 3.11 + +.. only:: html + + Build a virtual dataset (VRT). + +.. Index:: gdal raster buildvrt + +Synopsis +-------- + +.. code-block:: + + Usage: gdal raster buildvrt [OPTIONS] + + Build a virtual dataset (VRT). + + Positional arguments: + -i, --input Input raster datasets (or specify a @ to point to a file containing filenames) [1.. values] + -o, --output Output raster dataset (created by algorithm) [required] + + Common Options: + -h, --help Display help message and exit + --version Display GDAL version and exit + --json-usage Display usage as JSON document and exit + --drivers Display driver list as JSON document and exit + --progress Display progress bar + + Options: + --co, --creation-option = Creation option [may be repeated] + -b, --band Specify input band(s) number. [may be repeated] + --separate Place each input file into a separate band. + --overwrite Whether overwriting existing output is allowed + --resolution ,|average|highest|lowest> Target resolution (in destination CRS units) + --bbox Target bounding box (in destination CRS units) + --target-aligned-pixels Round target extent to target resolution + --srcnodata Set nodata values for input bands. [1.. values] + --vrtnodata Set nodata values at the VRT band level. [1.. values] + --hidenodata Makes the VRT band not report the NoData. + --addalpha Adds an alpha mask band to the VRT when the source raster have none. + + +Description +----------- + +This program builds a :ref:`VRT (Virtual Dataset) ` that is a mosaic of a list of +input GDAL datasets. The list of input GDAL datasets can be specified at the end +of the command line or put in a text file (one filename per line) for very long lists. +Wildcards '*', '?' or '['] of :cpp:func:`VSIGlob` can be used, even on files located +on network file systems such as /vsis3/, /vsigs/, /vsiaz/, etc. + +With :option:`--separate`, each input goes into a separate band in the VRT dataset. Otherwise, +the files are considered as source rasters of a larger mosaic and the VRT file has the same number of +bands as the input files. + +If one GDAL dataset is made of several subdatasets and has 0 raster bands, +all the subdatasets will be added to the VRT rather than the dataset itself. + +:program:`gdal raster buildvrt` does some checks to ensure that all files that will be put +in the resulting VRT have similar characteristics: number of bands, projection, color +interpretation, etc. If not, files that do not match the common characteristics will be skipped. +(This is only true in the default mode, and not when using the :option:`--separate` option) + +If the inputs spatially overlap, the order of the input list is used to determine priority. +Files that are listed at the end are the ones +from which the content will be fetched. Note that nodata will be taken into account +to potentially fetch data from lower-priority datasets, but currently, alpha channel +is not taken into account to do alpha compositing (so a source with alpha=0 +appearing on top of another source will override its content). This might be +changed in later versions. + +The following options are available: + +.. option:: -b + + Select an input to be processed. Bands are numbered from 1. + If input bands not set all bands will be added to the VRT. + Multiple :option:`-b` switches may be used to select a set of input bands. + +.. option:: --separate + + Place each input file into a separate band. See :example:`separate`. + Contrary to the default mode, it is not + required that all bands have the same datatype. + + All bands of each input file are added as separate + VRT bands, unless :option:`-b` is specified to select a subset of them. + +.. option:: --resolution {|highest|lowest|average} + + In case the resolution of all input files is not the same, the :option:`--resolution` flag + enables the user to control the way the output resolution is computed. + + `highest` will pick the smallest values of pixel dimensions within the set of source rasters. + + `lowest` will pick the largest values of pixel dimensions within the set of source rasters. + + `average` is the default and will compute an average of pixel dimensions within the set of source rasters. + + ,. The values must be expressed in georeferenced units. + Both must be positive values. + +.. option:: --bbox ,,, + + Set georeferenced extents of VRT file. The values must be expressed in georeferenced units. + If not specified, the extent of the VRT is the minimum bounding box of the set of source rasters. + Pixels within the extent of the VRT but not covered by a source raster will be read as valid + pixels with a value of zero unless a NODATA value is specified using :option:`--vrtnodata` + or an alpha mask band is added with :option:`--addalpha`. + +.. option:: --target-aligned-pixels + + (target aligned pixels) align + the coordinates of the extent of the output file to the values of the :option:`--resolution`, + such that the aligned extent includes the minimum extent. + Alignment means that xmin / resx, ymin / resy, xmax / resx and ymax / resy are integer values. + +.. option:: --srcnodata [,]... + + Set nodata values for input bands (different values can be supplied for each band). + If the option is not specified, the intrinsic nodata settings on the source datasets + will be used (if they exist). The value set by this option is written in the NODATA element + of each ``ComplexSource`` element. + +.. option:: --vrtnodata [,]... + + Set nodata values at the VRT band level (different values can be supplied for each band). If more + than one value is supplied, all values should be quoted to keep them together + as a single operating system argument (:example:`vrtnodata`). If the option is not specified, + intrinsic nodata settings on the first dataset will be used (if they exist). The value set by this option + is written in the ``NoDataValue`` element of each ``VRTRasterBand element``. Use a value of + `None` to ignore intrinsic nodata settings on the source datasets. + +.. option:: --addalpha + + Adds an alpha mask band to the VRT when the source raster have none. Mainly useful for RGB sources (or grey-level sources). + The alpha band is filled on-the-fly with the value 0 in areas without any source raster, and with value + 255 in areas with source raster. The effect is that a RGBA viewer will render + the areas without source rasters as transparent and areas with source rasters as opaque. + This option is not compatible with :option:`--separate`. + +.. option:: --hidenodata + + Even if any band contains nodata value, giving this option makes the VRT band + not report the NoData. Useful when you want to control the background color of + the dataset. By using along with the -addalpha option, you can prepare a + dataset which doesn't report nodata value but is transparent in areas with no + data. + + +Examples +-------- + +.. example:: + :title: Make a RGB virtual mosaic from 3 single-band input files + :id: separate + + .. code-block:: bash + + gdal raster biuldvrt --separate red.tif green.tif blue.tif rgb.vrt + +.. example:: + :title: Make a virtual mosaic with blue background colour (RGB: 0 0 255) + :id: vrtnodata + + .. code-block:: bash + + gdal raster buildvrt --hidenodata --vrtnodata=0,0,255 doq/*.tif doq_index.vrt + + +.. below is an allow-list for spelling checker. + +.. spelling:word-list:: + buildvrt + diff --git a/doc/source/programs/index.rst b/doc/source/programs/index.rst index aa72042f6559..c2160d4b7ff5 100644 --- a/doc/source/programs/index.rst +++ b/doc/source/programs/index.rst @@ -29,6 +29,7 @@ single :program:`gdal` program that accepts commands and subcommands. gdal_info gdal_convert gdal_raster + gdal_raster_buildvrt gdal_raster_info gdal_raster_convert gdal_raster_edit @@ -45,6 +46,7 @@ single :program:`gdal` program that accepts commands and subcommands. - :ref:`gdal_info_command`: Get information on a dataset - :ref:`gdal_convert_command`: Convert a dataset - :ref:`gdal_raster_command`: Entry point for raster commands + - :ref:`gdal_raster_buildvrt_subcommand`: buildvrt: Build a virtual dataset (VRT) - :ref:`gdal_raster_info_subcommand`: Get information on a raster dataset - :ref:`gdal_raster_convert_subcommand`: Convert a raster dataset - :ref:`gdal_raster_edit_subcommand`: Edit in place a raster dataset diff --git a/doc/source/user/virtual_file_systems.rst b/doc/source/user/virtual_file_systems.rst index 1b1eb0526dd2..ddacadfc5b0c 100644 --- a/doc/source/user/virtual_file_systems.rst +++ b/doc/source/user/virtual_file_systems.rst @@ -387,6 +387,7 @@ Starting with GDAL 2.3, options can be passed in the filename with the following - referer=value: HTTP Referer header - cookie=value: HTTP Cookie header - header_file=value: Filename that contains one or several "Header: Value" lines +- header.=: HTTP request header of name and value . (GDAL >= 3.11). e.g. ``header.Accept=application%2Fjson`` - unsafessl=yes/no - low_speed_time=value - low_speed_limit=value diff --git a/docker/alpine-normal/Dockerfile b/docker/alpine-normal/Dockerfile index 0582d83d297c..fa53d5d639c7 100644 --- a/docker/alpine-normal/Dockerfile +++ b/docker/alpine-normal/Dockerfile @@ -323,6 +323,7 @@ RUN date RUN apk add --no-cache \ armadillo \ + bash-completion \ basisu \ blosc \ brotli-libs \ @@ -413,7 +414,10 @@ COPY --from=builder /build_proj/usr/include/ /usr/include/ COPY --from=builder /build_proj/usr/bin/ /usr/bin/ COPY --from=builder /build_proj/usr/lib/ /usr/lib/ +COPY --from=builder /build/usr/share/bash-completion/ /usr/share/bash-completion/ COPY --from=builder /build/usr/share/gdal/ /usr/share/gdal/ COPY --from=builder /build/usr/include/ /usr/include/ COPY --from=builder /build_gdal_python/usr/ /usr/ COPY --from=builder /build_gdal_version_changing/usr/ /usr/ + +CMD ["/bin/bash", "-l"] diff --git a/docker/alpine-small/Dockerfile b/docker/alpine-small/Dockerfile index 1add1c3d1c01..88ffca12b34d 100644 --- a/docker/alpine-small/Dockerfile +++ b/docker/alpine-small/Dockerfile @@ -172,6 +172,7 @@ RUN date RUN apk add --no-cache \ libstdc++ \ + bash-completion \ sqlite-libs \ libcurl tiff \ zlib zstd-libs lz4-libs libdeflate libarchive \ @@ -191,6 +192,9 @@ COPY --from=builder /build_proj/usr/include/ /usr/include/ COPY --from=builder /build_proj/usr/bin/ /usr/bin/ COPY --from=builder /build_proj/usr/lib/ /usr/lib/ +COPY --from=builder /build/usr/share/bash-completion/ /usr/share/bash-completion/ COPY --from=builder /build/usr/share/gdal/ /usr/share/gdal/ COPY --from=builder /build/usr/include/ /usr/include/ COPY --from=builder /build_gdal_version_changing/usr/ /usr/ + +CMD ["/bin/bash", "-l"] diff --git a/docker/ubuntu-full/Dockerfile b/docker/ubuntu-full/Dockerfile index e0d9383ac262..d6b6821c37b9 100644 --- a/docker/ubuntu-full/Dockerfile +++ b/docker/ubuntu-full/Dockerfile @@ -527,7 +527,7 @@ RUN --mount=type=cache,target=/var/cache/apt,sharing=locked \ wget curl unzip ca-certificates \ # GDAL dependencies && apt-get install -y \ - libopenjp2-7 libcairo2 python3-numpy \ + bash-completion libopenjp2-7 libcairo2 python3-numpy \ libpng16-16 libjpeg-turbo8 libgif7 liblzma5 libgeos3.12.1 libgeos-c1v5 \ libxml2 libexpat1 \ libxerces-c3.2 libnetcdf-c++4-1 netcdf-bin libpoppler134 libspatialite8 librasterlite2-1 gpsbabel \ @@ -606,6 +606,7 @@ COPY --from=builder /build${PROJ_INSTALL_PREFIX}/include/ ${PROJ_INSTALL_PREFIX COPY --from=builder /build${PROJ_INSTALL_PREFIX}/bin/ ${PROJ_INSTALL_PREFIX}/bin/ COPY --from=builder /build${PROJ_INSTALL_PREFIX}/lib/ ${PROJ_INSTALL_PREFIX}/lib/ +COPY --from=builder /build/usr/share/bash-completion/ /usr/share/bash-completion/ COPY --from=builder /build/usr/share/java /usr/share/java COPY --from=builder /build/usr/share/gdal/ /usr/share/gdal/ COPY --from=builder /build/usr/include/ /usr/include/ @@ -618,3 +619,7 @@ RUN ldconfig RUN if test "$(uname -p)" = "x86_64"; then \ ogrinfo ADBC::memory: -oo ADBC_DRIVER=duckdb -oo PRELUDE_STATEMENTS="INSTALL spatial"; \ fi + +RUN echo "source /usr/share/bash-completion/bash_completion" >> /root/.bashrc + +CMD ["/bin/bash", "-l"] diff --git a/docker/ubuntu-small/Dockerfile b/docker/ubuntu-small/Dockerfile index 65cf4fd17b3a..8ec7c062af66 100644 --- a/docker/ubuntu-small/Dockerfile +++ b/docker/ubuntu-small/Dockerfile @@ -235,7 +235,7 @@ RUN --mount=type=cache,target=/var/cache/apt,sharing=locked \ curl unzip ca-certificates \ # GDAL dependencies && apt-get install -y --no-install-recommends \ - python3-numpy libpython3.12 \ + bash-completion python3-numpy libpython3.12 \ libjpeg-turbo8 libgeos3.12.1 libgeos-c1v5 \ libexpat1 \ libxerces-c3.2 \ @@ -258,9 +258,14 @@ COPY --from=builder /build${PROJ_INSTALL_PREFIX}/include/ ${PROJ_INSTALL_PREFIX COPY --from=builder /build${PROJ_INSTALL_PREFIX}/bin/ ${PROJ_INSTALL_PREFIX}/bin/ COPY --from=builder /build${PROJ_INSTALL_PREFIX}/lib/ ${PROJ_INSTALL_PREFIX}/lib/ +COPY --from=builder /build/usr/share/bash-completion/ /usr/share/bash-completion/ COPY --from=builder /build/usr/share/gdal/ /usr/share/gdal/ COPY --from=builder /build/usr/include/ /usr/include/ COPY --from=builder /build_gdal_python/usr/ /usr/ COPY --from=builder /build_gdal_version_changing/usr/ /usr/ RUN ldconfig + +RUN echo "source /usr/share/bash-completion/bash_completion" >> /root/.bashrc + +CMD ["/bin/bash", "-l"] diff --git a/frmts/CMakeLists.txt b/frmts/CMakeLists.txt index 23a5a3a78607..a484587b6d55 100644 --- a/frmts/CMakeLists.txt +++ b/frmts/CMakeLists.txt @@ -79,6 +79,7 @@ if (CMAKE_BUILD_TYPE MATCHES "Debug" OR GDAL_ENABLE_DRIVER_NULL) gdal_optional_format(null "NULL dummy driver") endif () +gdal_optional_format(libertiff "GeoTIFF support through libertiff library") gdal_optional_format(hfa "Erdas Imagine .img") gdal_optional_format(sdts "SDTS translator") gdal_optional_format(nitf "National Imagery Transmission Format") diff --git a/frmts/drivers.ini b/frmts/drivers.ini index 11d381945755..c28e48407382 100644 --- a/frmts/drivers.ini +++ b/frmts/drivers.ini @@ -17,6 +17,7 @@ GTI SNAP_TIFF GTiff COG +LIBERTIFF NITF RPFTOC ECRGTOC diff --git a/frmts/gdalallregister.cpp b/frmts/gdalallregister.cpp index 005fd734ce89..a8f073396e17 100644 --- a/frmts/gdalallregister.cpp +++ b/frmts/gdalallregister.cpp @@ -332,6 +332,10 @@ void CPL_STDCALL GDALAllRegister() GDALRegister_COG(); #endif +#ifdef FRMT_libertiff + GDALRegister_LIBERTIFF(); +#endif + #ifdef FRMT_nitf GDALRegister_NITF(); GDALRegister_RPFTOC(); diff --git a/frmts/gtiff/cogdriver.cpp b/frmts/gtiff/cogdriver.cpp index ac56ca0d2845..310204a64127 100644 --- a/frmts/gtiff/cogdriver.cpp +++ b/frmts/gtiff/cogdriver.cpp @@ -1481,10 +1481,12 @@ void GDALCOGDriver::InitializeCreationOptionList() osOptions += " " "