LCOV - code coverage report
Current view: top level - frmts/snap_tiff - snaptiffdriver.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 321 345 93.0 %
Date: 2025-01-18 12:42:00 Functions: 18 18 100.0 %

          Line data    Source code
       1             : // SPDX-License-Identifier: MIT
       2             : // Copyright 2024, Even Rouault <even.rouault at spatialys.com>
       3             : 
       4             : #include "cpl_port.h"
       5             : #include "cpl_minixml.h"
       6             : #include "cpl_vsi_virtual.h"
       7             : #include "gdal_pam.h"
       8             : #include "rawdataset.h"
       9             : 
      10             : #define LIBERTIFF_NS GDAL_libertiff
      11             : #include "../../third_party/libertiff/libertiff.hpp"
      12             : 
      13             : #include <algorithm>
      14             : #include <cmath>
      15             : 
      16             : constexpr const char *SNAP_TIFF_PREFIX = "SNAP_TIFF:";
      17             : 
      18             : // Non-standard TIFF tag holding DIMAP XML for SNAP TIFF products
      19             : constexpr uint16_t DIMAP_TAG = 65000;
      20             : 
      21             : // Number of values per GCP in the GeoTIFFTiePoints tag
      22             : constexpr int VALUES_PER_GCP = 6;
      23             : 
      24             : constexpr int GCP_PIXEL = 0;
      25             : constexpr int GCP_LINE = 1;
      26             : // constexpr int GCP_DEPTH = 2;
      27             : constexpr int GCP_X = 3;
      28             : constexpr int GCP_Y = 4;
      29             : constexpr int GCP_Z = 5;
      30             : 
      31             : /************************************************************************/
      32             : /*                            SNAPTIFFDataset                           */
      33             : /************************************************************************/
      34             : 
      35             : class SNAPTIFFDataset final : public GDALPamDataset
      36             : {
      37             :   public:
      38           3 :     SNAPTIFFDataset() = default;
      39             : 
      40             :     static int Identify(GDALOpenInfo *poOpenInfo);
      41             :     static GDALDataset *Open(GDALOpenInfo *poOpenInfo);
      42             : 
      43             :     char **GetMetadataDomainList() override;
      44             :     char **GetMetadata(const char *pszDomain = "") override;
      45             :     const char *GetMetadataItem(const char *pszName,
      46             :                                 const char *pszDomain = "") override;
      47             : 
      48           1 :     const OGRSpatialReference *GetGCPSpatialRef() const override
      49             :     {
      50           1 :         return m_oSRS.IsEmpty() ? nullptr : &m_oSRS;
      51             :     }
      52             : 
      53           2 :     int GetGCPCount() override
      54             :     {
      55           2 :         return int(m_asGCPs.size());
      56             :     }
      57             : 
      58           1 :     const GDAL_GCP *GetGCPs() override
      59             :     {
      60           1 :         return gdal::GCP::c_ptr(m_asGCPs);
      61             :     }
      62             : 
      63             :   private:
      64             :     VSIVirtualHandleUniquePtr m_poFile{};
      65             :     std::unique_ptr<const LIBERTIFF_NS::Image> m_poImage{};
      66             : 
      67             :     //! Whether this dataset is actually the geolocation array
      68             :     bool m_bIsGeolocArray = false;
      69             : 
      70             :     //! Content of "xml:DIMAP" metadata domain
      71             :     CPLStringList m_aosDIMAPMetadata{};
      72             : 
      73             :     CPLStringList m_aosGEOLOCATION{};
      74             :     int m_nGeolocArrayWidth = 0;
      75             :     int m_nGeolocArrayHeight = 0;
      76             : 
      77             :     CPLStringList m_aosSUBDATASETS{};
      78             : 
      79             :     std::vector<gdal::GCP> m_asGCPs{};
      80             :     OGRSpatialReference m_oSRS{};
      81             : 
      82             :     bool GetGeolocationMetadata();
      83             : 
      84             :     void ReadSRS();
      85             : };
      86             : 
      87             : /************************************************************************/
      88             : /*                              Identify()                              */
      89             : /************************************************************************/
      90             : 
      91       76662 : int SNAPTIFFDataset::Identify(GDALOpenInfo *poOpenInfo)
      92             : {
      93       76662 :     if (STARTS_WITH(poOpenInfo->pszFilename, SNAP_TIFF_PREFIX))
      94          12 :         return true;
      95             : 
      96       76650 :     if (poOpenInfo->fpL == nullptr || poOpenInfo->nHeaderBytes < 16 ||
      97             :         // BigEndian classic TIFF
      98       26789 :         memcmp(poOpenInfo->pabyHeader, "\x4D\x4D\x00\x2A", 4) != 0)
      99             :     {
     100       76518 :         return false;
     101             :     }
     102             : 
     103             :     struct MyFileReader final : public LIBERTIFF_NS::FileReader
     104             :     {
     105             :         const GDALOpenInfo *m_poOpenInfo;
     106             : 
     107         132 :         explicit MyFileReader(const GDALOpenInfo *poOpenInfoIn)
     108         132 :             : m_poOpenInfo(poOpenInfoIn)
     109             :         {
     110         132 :         }
     111             : 
     112         136 :         uint64_t size() const override
     113             :         {
     114         136 :             return m_poOpenInfo->nHeaderBytes;
     115             :         }
     116             : 
     117        6875 :         size_t read(uint64_t offset, size_t count, void *buffer) const override
     118             :         {
     119        6875 :             if (offset + count >
     120        6875 :                 static_cast<uint64_t>(m_poOpenInfo->nHeaderBytes))
     121         154 :                 return 0;
     122        6721 :             memcpy(buffer,
     123        6721 :                    m_poOpenInfo->pabyHeader + static_cast<size_t>(offset),
     124             :                    count);
     125        6721 :             return count;
     126             :         }
     127             : 
     128             :         CPL_DISALLOW_COPY_ASSIGN(MyFileReader)
     129             :     };
     130             : 
     131         264 :     auto f = std::make_shared<const MyFileReader>(poOpenInfo);
     132             : #ifdef DEBUG
     133             :     // Just to increase coverage testing
     134         132 :     CPLAssert(f->size() == uint64_t(poOpenInfo->nHeaderBytes));
     135             :     char dummy;
     136         132 :     CPLAssert(f->read(poOpenInfo->nHeaderBytes, 1, &dummy) == 0);
     137             : #endif
     138         264 :     auto image = LIBERTIFF_NS::open</*acceptBigTIFF = */ false>(std::move(f));
     139             :     // Checks that it is a single-band Float32 uncompressed dataset, made
     140             :     // of a single strip.
     141         339 :     if (!image || image->nextImageOffset() != 0 ||
     142         185 :         image->compression() != LIBERTIFF_NS::Compression::None ||
     143          98 :         image->sampleFormat() != LIBERTIFF_NS::SampleFormat::IEEEFP ||
     144          23 :         image->samplesPerPixel() != 1 || image->bitsPerSample() != 32 ||
     145          18 :         image->isTiled() || image->strileCount() != 1 || image->width() == 0 ||
     146          12 :         image->width() > INT_MAX / sizeof(float) || image->height() == 0 ||
     147          12 :         image->height() > INT_MAX || image->rowsPerStrip() != image->height() ||
     148          12 :         !image->tag(LIBERTIFF_NS::TagCode::GeoTIFFPixelScale) ||
     149          12 :         !image->tag(LIBERTIFF_NS::TagCode::GeoTIFFTiePoints) ||
     150         249 :         !image->tag(LIBERTIFF_NS::TagCode::GeoTIFFGeoKeyDirectory) ||
     151           6 :         !image->tag(DIMAP_TAG))
     152             :     {
     153         128 :         return false;
     154             :     }
     155             : 
     156           4 :     return true;
     157             : }
     158             : 
     159             : /************************************************************************/
     160             : /*                               Open()                                 */
     161             : /************************************************************************/
     162             : 
     163           7 : GDALDataset *SNAPTIFFDataset::Open(GDALOpenInfo *poOpenInfo)
     164             : {
     165           7 :     if (poOpenInfo->eAccess == GA_Update || !Identify(poOpenInfo))
     166           0 :         return nullptr;
     167             : 
     168           7 :     bool bIsGeolocation = false;
     169             :     // Check if it is SNAP_TIFF:"filename":{subdataset_component} syntax
     170           7 :     if (STARTS_WITH(poOpenInfo->pszFilename, SNAP_TIFF_PREFIX))
     171             :     {
     172             :         const CPLStringList aosTokens(CSLTokenizeString2(
     173           6 :             poOpenInfo->pszFilename, ":", CSLT_HONOURSTRINGS));
     174           6 :         if (aosTokens.size() != 3)
     175           1 :             return nullptr;
     176           5 :         bIsGeolocation = EQUAL(aosTokens[2], "GEOLOCATION");
     177           5 :         if (!bIsGeolocation && !EQUAL(aosTokens[2], "MAIN"))
     178           1 :             return nullptr;
     179           4 :         GDALOpenInfo oOpenInfo(aosTokens[1], GA_ReadOnly);
     180           4 :         if (!Identify(&oOpenInfo))
     181           2 :             return nullptr;
     182           2 :         std::swap(poOpenInfo->fpL, oOpenInfo.fpL);
     183           2 :         if (!poOpenInfo->fpL)
     184           0 :             return nullptr;
     185             :     }
     186             : 
     187             :     struct MyFileReader final : public LIBERTIFF_NS::FileReader
     188             :     {
     189             :         VSILFILE *m_fp;
     190             :         mutable uint64_t m_nFileSize = 0;
     191             : 
     192           3 :         explicit MyFileReader(VSILFILE *fp) : m_fp(fp)
     193             :         {
     194           3 :         }
     195             : 
     196           9 :         uint64_t size() const override
     197             :         {
     198           9 :             if (m_nFileSize == 0)
     199             :             {
     200           3 :                 m_fp->Seek(0, SEEK_END);
     201           3 :                 m_nFileSize = m_fp->Tell();
     202             :             }
     203           9 :             return m_nFileSize;
     204             :         }
     205             : 
     206         261 :         size_t read(uint64_t offset, size_t count, void *buffer) const override
     207             :         {
     208         261 :             return m_fp->Seek(offset, SEEK_SET) == 0
     209         261 :                        ? m_fp->Read(buffer, 1, count)
     210         261 :                        : 0;
     211             :         }
     212             : 
     213             :         CPL_DISALLOW_COPY_ASSIGN(MyFileReader)
     214             :     };
     215             : 
     216           6 :     auto f = std::make_shared<const MyFileReader>(poOpenInfo->fpL);
     217             : #ifdef DEBUG
     218             :     // Just to increase coverage testing
     219             :     char dummy;
     220           3 :     CPLAssert(f->read(f->size(), 1, &dummy) == 0);
     221             : #endif
     222             : 
     223           6 :     auto poDS = std::make_unique<SNAPTIFFDataset>();
     224             : 
     225           3 :     poDS->m_poImage = LIBERTIFF_NS::open(std::move(f));
     226           3 :     if (!poDS->m_poImage)
     227           0 :         return nullptr;
     228           3 :     poDS->m_poFile.reset(poOpenInfo->fpL);
     229           3 :     poOpenInfo->fpL = nullptr;
     230             : 
     231           3 :     poDS->nRasterXSize = static_cast<int>(poDS->m_poImage->width());
     232           3 :     poDS->nRasterYSize = static_cast<int>(poDS->m_poImage->height());
     233           3 :     poDS->SetDescription(poOpenInfo->pszFilename);
     234             : 
     235           3 :     if (bIsGeolocation)
     236             :     {
     237           1 :         poDS->m_bIsGeolocArray = true;
     238           1 :         if (!poDS->GetGeolocationMetadata())
     239             :         {
     240           0 :             return nullptr;
     241             :         }
     242           1 :         poDS->nRasterXSize = poDS->m_nGeolocArrayWidth;
     243           1 :         poDS->nRasterYSize = poDS->m_nGeolocArrayHeight;
     244             : 
     245             :         const auto *psTag =
     246           1 :             poDS->m_poImage->tag(LIBERTIFF_NS::TagCode::GeoTIFFTiePoints);
     247             : 
     248           3 :         for (int iBand = 1; iBand <= 2; ++iBand)
     249             :         {
     250             :             auto poBand = std::make_unique<RawRasterBand>(
     251           2 :                 poDS->m_poFile.get(),
     252           2 :                 psTag->value_offset +
     253           2 :                     (iBand == 1 ? GCP_X : GCP_Y) * sizeof(double),
     254           0 :                 int(sizeof(double)) * VALUES_PER_GCP,
     255           2 :                 int(sizeof(double)) * VALUES_PER_GCP * poDS->nRasterXSize,
     256           2 :                 GDT_Float64, !poDS->m_poImage->mustByteSwap(),
     257           2 :                 poDS->nRasterXSize, poDS->nRasterYSize,
     258           4 :                 RawRasterBand::OwnFP::NO);
     259           2 :             if (!poBand->IsValid())
     260           0 :                 return nullptr;
     261           2 :             if (iBand == 1)
     262           1 :                 poBand->SetDescription("Longitude");
     263             :             else
     264           1 :                 poBand->SetDescription("Latitude");
     265           2 :             poDS->SetBand(iBand, std::move(poBand));
     266             :         }
     267             : 
     268           1 :         return poDS.release();
     269             :     }
     270             : 
     271           2 :     poDS->ReadSRS();
     272           2 :     poDS->GetGeolocationMetadata();
     273             : 
     274             :     {
     275           2 :         bool okStrileOffset = false;
     276             :         auto poBand = std::make_unique<RawRasterBand>(
     277           2 :             poDS->m_poFile.get(),
     278           2 :             poDS->m_poImage->strileOffset(0, okStrileOffset),
     279           2 :             int(sizeof(float)), int(sizeof(float)) * poDS->nRasterXSize,
     280           2 :             GDT_Float32, !poDS->m_poImage->mustByteSwap(), poDS->nRasterXSize,
     281           4 :             poDS->nRasterYSize, RawRasterBand::OwnFP::NO);
     282           2 :         if (!poBand->IsValid())
     283           0 :             return nullptr;
     284           2 :         poDS->SetBand(1, std::move(poBand));
     285             :     }
     286           2 :     auto poBand = poDS->papoBands[0];
     287             : 
     288             :     const auto &psImageDescription =
     289           2 :         poDS->m_poImage->tag(LIBERTIFF_NS::TagCode::ImageDescription);
     290           2 :     if (psImageDescription &&
     291           2 :         psImageDescription->type == LIBERTIFF_NS::TagType::ASCII &&
     292           2 :         !psImageDescription->invalid_value_offset &&
     293             :         // Sanity check
     294           2 :         psImageDescription->count < 100 * 1000)
     295             :     {
     296           2 :         bool ok = true;
     297             :         const std::string s =
     298           4 :             poDS->m_poImage->readTagAsString(*psImageDescription, ok);
     299           2 :         if (ok)
     300             :         {
     301           2 :             poDS->GDALDataset::SetMetadataItem("IMAGE_DESCRIPTION", s.c_str());
     302             :         }
     303             :     }
     304             : 
     305           2 :     const auto *psDimapTag = poDS->m_poImage->tag(DIMAP_TAG);
     306           2 :     if (psDimapTag && psDimapTag->type == LIBERTIFF_NS::TagType::ASCII &&
     307           2 :         !psDimapTag->invalid_value_offset)
     308             :     {
     309             :         try
     310             :         {
     311           2 :             bool ok = true;
     312             :             // Just read the first 10 kB max to fetch essential band metadata
     313           2 :             const std::string s = poDS->m_poImage->readContext()->readString(
     314           2 :                 psDimapTag->value_offset,
     315             :                 static_cast<size_t>(
     316           2 :                     std::min<uint64_t>(psDimapTag->count, 10000)),
     317           6 :                 ok);
     318           2 :             const auto posStart = s.find("<Spectral_Band_Info>");
     319           2 :             if (posStart != std::string::npos)
     320             :             {
     321           2 :                 const char *markerEnd = "</Spectral_Band_Info>";
     322           2 :                 const auto posEnd = s.find(markerEnd, posStart);
     323           2 :                 if (posEnd != std::string::npos)
     324             :                 {
     325             :                     const std::string osSubstr = s.substr(
     326           4 :                         posStart, posEnd - posStart + strlen(markerEnd));
     327           4 :                     CPLXMLTreeCloser oRoot(CPLParseXMLString(osSubstr.c_str()));
     328           2 :                     if (oRoot)
     329             :                     {
     330           2 :                         const char *pszNoDataValueUsed = CPLGetXMLValue(
     331           2 :                             oRoot.get(), "NO_DATA_VALUE_USED", nullptr);
     332           2 :                         const char *pszNoDataValue = CPLGetXMLValue(
     333           2 :                             oRoot.get(), "NO_DATA_VALUE", nullptr);
     334           4 :                         if (pszNoDataValueUsed && pszNoDataValue &&
     335           2 :                             CPLTestBool(pszNoDataValueUsed))
     336             :                         {
     337           2 :                             poBand->SetNoDataValue(CPLAtof(pszNoDataValue));
     338             :                         }
     339             : 
     340           2 :                         if (const char *pszScalingFactor = CPLGetXMLValue(
     341           2 :                                 oRoot.get(), "SCALING_FACTOR", nullptr))
     342             :                         {
     343           2 :                             poBand->SetScale(CPLAtof(pszScalingFactor));
     344             :                         }
     345             : 
     346           2 :                         if (const char *pszScalingOffset = CPLGetXMLValue(
     347           2 :                                 oRoot.get(), "SCALING_OFFSET", nullptr))
     348             :                         {
     349           2 :                             poBand->SetOffset(CPLAtof(pszScalingOffset));
     350             :                         }
     351             : 
     352           2 :                         if (const char *pszBandName = CPLGetXMLValue(
     353           2 :                                 oRoot.get(), "BAND_NAME", nullptr))
     354             :                         {
     355           2 :                             poBand->SetDescription(pszBandName);
     356             :                         }
     357             : 
     358           2 :                         if (const char *pszUnit = CPLGetXMLValue(
     359           2 :                                 oRoot.get(), "PHYSICAL_UNIT", nullptr))
     360             :                         {
     361           2 :                             poBand->SetUnitType(pszUnit);
     362             :                         }
     363             :                     }
     364             :                 }
     365             :             }
     366             :         }
     367           0 :         catch (const std::exception &)
     368             :         {
     369             :         }
     370             :     }
     371             : 
     372             :     // Initialize PAM
     373           2 :     poDS->SetDescription(poOpenInfo->pszFilename);
     374           2 :     poDS->TryLoadXML();
     375             : 
     376             :     // Check for overviews.
     377           2 :     poDS->oOvManager.Initialize(poDS.get(), poOpenInfo->pszFilename);
     378             : 
     379           2 :     return poDS.release();
     380             : }
     381             : 
     382             : /************************************************************************/
     383             : /*                      GetMetadataDomainList()                         */
     384             : /************************************************************************/
     385             : 
     386           1 : char **SNAPTIFFDataset::GetMetadataDomainList()
     387             : {
     388           1 :     return BuildMetadataDomainList(GDALPamDataset::GetMetadataDomainList(),
     389             :                                    TRUE, "GEOLOCATION", "SUBDATASETS",
     390           1 :                                    "xml:DIMAP", nullptr);
     391             : }
     392             : 
     393             : /************************************************************************/
     394             : /*                      GetGeolocationMetadata()                        */
     395             : /************************************************************************/
     396             : 
     397             : // (Partially) read the content of the GeoTIFFTiePoints tag to check if the
     398             : // tie points form a regular geolocation array, and extract the width, height,
     399             : // and spacing of that geolocation array. Also fills the m_aosGEOLOCATION
     400             : // metadata domain
     401           4 : bool SNAPTIFFDataset::GetGeolocationMetadata()
     402             : {
     403           4 :     const auto *psTag = m_poImage->tag(LIBERTIFF_NS::TagCode::GeoTIFFTiePoints);
     404           4 :     if (psTag && psTag->type == LIBERTIFF_NS::TagType::Double &&
     405           4 :         !psTag->invalid_value_offset && (psTag->count % VALUES_PER_GCP) == 0 &&
     406             :         // Sanity check
     407           4 :         psTag->count <= uint64_t(nRasterXSize) * nRasterYSize * VALUES_PER_GCP)
     408             :     {
     409           4 :         const int numGCPs = int(psTag->count / VALUES_PER_GCP);
     410             : 
     411             :         // Assume that the geolocation array has the same proportion as the
     412             :         // main array
     413             :         const double dfGCPArrayWidth =
     414           4 :             sqrt(double(nRasterXSize) * numGCPs / nRasterYSize);
     415             :         const double dfGCPArrayHeight =
     416           4 :             sqrt(double(nRasterYSize) * numGCPs / nRasterXSize);
     417           4 :         if (dfGCPArrayWidth > INT_MAX || dfGCPArrayHeight > INT_MAX)
     418             :         {
     419           0 :             return false;
     420             :         }
     421             : 
     422           4 :         const int nGCPArrayWidth = int(std::round(dfGCPArrayWidth));
     423           4 :         const int nGCPArrayHeight = int(std::round(dfGCPArrayHeight));
     424           4 :         constexpr int NUM_LINES = 3;
     425           4 :         if (nGCPArrayWidth * nGCPArrayHeight != numGCPs ||
     426             :             nGCPArrayHeight < NUM_LINES)
     427             :         {
     428           0 :             return false;
     429             :         }
     430             : 
     431           4 :         bool ok = true;
     432             :         // Just read the first 3 lines of the geolocation array
     433           4 :         const auto numValuesPerLine = nGCPArrayWidth * VALUES_PER_GCP;
     434           4 :         const auto values = m_poImage->readContext()->readArray<double>(
     435           8 :             psTag->value_offset, numValuesPerLine * NUM_LINES, ok);
     436           4 :         if (!ok || values.empty())
     437           0 :             return false;
     438             : 
     439           4 :         if (values[GCP_LINE] != 0.5 && values[GCP_PIXEL] != 0.5)
     440           0 :             return false;
     441             : 
     442           4 :         constexpr double RELATIVE_TOLERANCE = 1e-5;
     443           4 :         constexpr double PIXEL_TOLERANCE = 1e-3;
     444             : 
     445             :         // Check that pixel spacing is constant on all three first lines
     446             :         const double pixelSpacing =
     447           4 :             values[VALUES_PER_GCP + GCP_PIXEL] - values[GCP_PIXEL];
     448           4 :         if (!(pixelSpacing >= 1))
     449           0 :             return false;
     450           4 :         if (std::fabs(pixelSpacing * (nGCPArrayWidth - 1) -
     451           4 :                       (nRasterXSize - 1)) > PIXEL_TOLERANCE)
     452           0 :             return false;
     453             : 
     454             :         double adfY[NUM_LINES];
     455          16 :         for (int iLine = 0; iLine < NUM_LINES; ++iLine)
     456             :         {
     457          12 :             adfY[iLine] = values[iLine * numValuesPerLine + GCP_LINE];
     458          12 :             for (int i = iLine * numValuesPerLine + VALUES_PER_GCP;
     459       19140 :                  i < (iLine + 1) * numValuesPerLine; i += VALUES_PER_GCP)
     460             :             {
     461       38256 :                 if (values[i + GCP_LINE] !=
     462       19128 :                     values[i - VALUES_PER_GCP + GCP_LINE])
     463             :                 {
     464           0 :                     return false;
     465             :                 }
     466             :                 const double pixelSpacingNew =
     467       19128 :                     values[i + GCP_PIXEL] -
     468       19128 :                     values[i - VALUES_PER_GCP + GCP_PIXEL];
     469       19128 :                 if (std::fabs(pixelSpacingNew - pixelSpacing) >
     470       19128 :                     RELATIVE_TOLERANCE * std::fabs(pixelSpacing))
     471             :                 {
     472           0 :                     return false;
     473             :                 }
     474             :             }
     475             :         }
     476             : 
     477             :         // Check that line spacing is constant on the three first lines
     478           4 :         const double lineSpacing = adfY[1] - adfY[0];
     479           4 :         if (!(lineSpacing >= 1))
     480           0 :             return false;
     481           4 :         if (std::fabs(lineSpacing * (nGCPArrayHeight - 1) -
     482           4 :                       (nRasterYSize - 1)) > PIXEL_TOLERANCE)
     483           0 :             return false;
     484             : 
     485           8 :         for (int iLine = 1; iLine + 1 < NUM_LINES; ++iLine)
     486             :         {
     487           4 :             const double lineSpacingNew = adfY[iLine + 1] - adfY[iLine];
     488           4 :             if (std::fabs(lineSpacingNew - lineSpacing) >
     489           4 :                 RELATIVE_TOLERANCE * std::fabs(lineSpacing))
     490             :             {
     491           0 :                 return false;
     492             :             }
     493             :         }
     494             : 
     495             :         // Read last line
     496           4 :         const auto lastLineValue = m_poImage->readContext()->readArray<double>(
     497           4 :             psTag->value_offset + uint64_t(nGCPArrayHeight - 1) *
     498           4 :                                       numValuesPerLine * sizeof(double),
     499           8 :             numValuesPerLine, ok);
     500           4 :         if (!ok)
     501           0 :             return false;
     502             : 
     503           4 :         if (!m_bIsGeolocArray)
     504             :         {
     505             :             // Expose the 4 corner GCPs for rough georeferencing.
     506             : 
     507           3 :             const uint32_t nShift = numValuesPerLine - VALUES_PER_GCP;
     508           3 :             m_asGCPs.emplace_back("TL", "Top Left", values[GCP_PIXEL],
     509           3 :                                   values[GCP_LINE], values[GCP_X],
     510           6 :                                   values[GCP_Y], values[GCP_Z]);
     511             :             m_asGCPs.emplace_back(
     512           3 :                 "TR", "Top Right", values[nShift + GCP_PIXEL],
     513           3 :                 values[nShift + GCP_LINE], values[nShift + GCP_X],
     514           6 :                 values[nShift + GCP_Y], values[nShift + GCP_Z]);
     515           3 :             m_asGCPs.emplace_back("BL", "Bottom Left", lastLineValue[GCP_PIXEL],
     516           3 :                                   lastLineValue[GCP_LINE], lastLineValue[GCP_X],
     517           6 :                                   lastLineValue[GCP_Y], lastLineValue[GCP_Z]);
     518             :             m_asGCPs.emplace_back(
     519           3 :                 "BR", "Bottom Right", lastLineValue[nShift + GCP_PIXEL],
     520           3 :                 lastLineValue[nShift + GCP_LINE], lastLineValue[nShift + GCP_X],
     521           6 :                 lastLineValue[nShift + GCP_Y], lastLineValue[nShift + GCP_Z]);
     522             :         }
     523             : 
     524           4 :         m_nGeolocArrayWidth = nGCPArrayWidth;
     525           4 :         m_nGeolocArrayHeight = nGCPArrayHeight;
     526             : 
     527           4 :         if (!m_bIsGeolocArray)
     528             :         {
     529           3 :             if (!m_oSRS.IsEmpty())
     530             :             {
     531           3 :                 char *pszWKT = nullptr;
     532           3 :                 m_oSRS.exportToWkt(&pszWKT);
     533           3 :                 m_aosGEOLOCATION.SetNameValue("SRS", pszWKT);
     534           3 :                 CPLFree(pszWKT);
     535             :             }
     536             : 
     537             :             m_aosGEOLOCATION.SetNameValue(
     538             :                 "X_DATASET", CPLSPrintf("%s\"%s\":GEOLOCATION",
     539           3 :                                         SNAP_TIFF_PREFIX, GetDescription()));
     540           3 :             m_aosGEOLOCATION.SetNameValue("X_BAND", "1");
     541             :             m_aosGEOLOCATION.SetNameValue(
     542             :                 "Y_DATASET", CPLSPrintf("%s\"%s\":GEOLOCATION",
     543           3 :                                         SNAP_TIFF_PREFIX, GetDescription()));
     544           3 :             m_aosGEOLOCATION.SetNameValue("Y_BAND", "2");
     545           3 :             m_aosGEOLOCATION.SetNameValue("PIXEL_OFFSET", "0");
     546             :             m_aosGEOLOCATION.SetNameValue("PIXEL_STEP",
     547           3 :                                           CPLSPrintf("%.17g", pixelSpacing));
     548           3 :             m_aosGEOLOCATION.SetNameValue("LINE_OFFSET", "0");
     549             :             m_aosGEOLOCATION.SetNameValue("LINE_STEP",
     550           3 :                                           CPLSPrintf("%.17g", lineSpacing));
     551             :         }
     552             : 
     553           4 :         return true;
     554             :     }
     555           0 :     return false;
     556             : }
     557             : 
     558             : /************************************************************************/
     559             : /*                             ReadSRS()                                */
     560             : /************************************************************************/
     561             : 
     562             : // Simplified GeoTIFF SRS reader, assuming the SRS is encoded as a EPSG code
     563           2 : void SNAPTIFFDataset::ReadSRS()
     564             : {
     565             :     const auto &psGeoKeysTag =
     566           2 :         m_poImage->tag(LIBERTIFF_NS::TagCode::GeoTIFFGeoKeyDirectory);
     567           2 :     constexpr int VALUES_PER_GEOKEY = 4;
     568           2 :     if (psGeoKeysTag && psGeoKeysTag->type == LIBERTIFF_NS::TagType::Short &&
     569           2 :         !psGeoKeysTag->invalid_value_offset &&
     570           2 :         psGeoKeysTag->count >= VALUES_PER_GEOKEY &&
     571           2 :         (psGeoKeysTag->count % VALUES_PER_GEOKEY) == 0 &&
     572             :         // Sanity check
     573           2 :         psGeoKeysTag->count < 1000)
     574             :     {
     575           2 :         bool ok = true;
     576             :         const auto values =
     577           4 :             m_poImage->readTagAsVector<uint16_t>(*psGeoKeysTag, ok);
     578           2 :         if (values.size() >= 4)
     579             :         {
     580           2 :             const uint16_t geokeysCount = values[3];
     581           2 :             constexpr uint16_t GEOTIFF_KEY_DIRECTORY_VERSION_V1 = 1;
     582           2 :             constexpr uint16_t GEOTIFF_KEY_VERSION_MAJOR_V1 = 1;
     583           2 :             if (values[0] == GEOTIFF_KEY_DIRECTORY_VERSION_V1 &&
     584             :                 // GeoTIFF 1.x
     585           4 :                 values[1] == GEOTIFF_KEY_VERSION_MAJOR_V1 &&
     586           2 :                 geokeysCount == psGeoKeysTag->count / VALUES_PER_GEOKEY - 1)
     587             :             {
     588           2 :                 constexpr uint16_t GeoTIFFTypeShort = 0;
     589           2 :                 constexpr uint16_t GeodeticCRSGeoKey = 2048;
     590           2 :                 constexpr uint16_t ProjectedCRSGeoKey = 3072;
     591           2 :                 uint16_t nEPSGCode = 0;
     592           8 :                 for (uint32_t i = 1; i <= geokeysCount; ++i)
     593             :                 {
     594           6 :                     const auto geokey = values[VALUES_PER_GEOKEY * i];
     595           6 :                     const auto geokeyType = values[VALUES_PER_GEOKEY * i + 1];
     596           6 :                     const auto geokeyCount = values[VALUES_PER_GEOKEY * i + 2];
     597           6 :                     const auto geokeyValue = values[VALUES_PER_GEOKEY * i + 3];
     598           6 :                     if ((geokey == GeodeticCRSGeoKey ||
     599           2 :                          geokey == ProjectedCRSGeoKey) &&
     600           2 :                         geokeyType == GeoTIFFTypeShort && geokeyCount == 1 &&
     601             :                         geokeyValue > 0)
     602             :                     {
     603           2 :                         nEPSGCode = geokeyValue;
     604           2 :                         if (geokey == ProjectedCRSGeoKey)
     605           0 :                             break;
     606             :                     }
     607             :                 }
     608           2 :                 if (nEPSGCode > 0)
     609             :                 {
     610           2 :                     m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
     611           2 :                     m_oSRS.importFromEPSG(nEPSGCode);
     612             :                 }
     613             :             }
     614             :         }
     615             :     }
     616           2 : }
     617             : 
     618             : /************************************************************************/
     619             : /*                            GetMetadata()                             */
     620             : /************************************************************************/
     621             : 
     622          11 : char **SNAPTIFFDataset::GetMetadata(const char *pszDomain)
     623             : {
     624          11 :     if (!m_bIsGeolocArray)
     625             :     {
     626          10 :         if (pszDomain && EQUAL(pszDomain, "xml:DIMAP"))
     627             :         {
     628           2 :             if (m_aosDIMAPMetadata.empty())
     629             :             {
     630           1 :                 const auto *psDimapTag = m_poImage->tag(DIMAP_TAG);
     631           1 :                 if (psDimapTag &&
     632           1 :                     psDimapTag->type == LIBERTIFF_NS::TagType::ASCII &&
     633           1 :                     !psDimapTag->invalid_value_offset &&
     634             :                     // Sanity check
     635           1 :                     psDimapTag->count < 100 * 1000 * 1000)
     636             :                 {
     637             :                     try
     638             :                     {
     639           1 :                         bool ok = true;
     640             :                         const std::string s =
     641           2 :                             m_poImage->readTagAsString(*psDimapTag, ok);
     642           1 :                         if (ok)
     643             :                         {
     644           1 :                             m_aosDIMAPMetadata.AddString(s.c_str());
     645             :                         }
     646             :                     }
     647           0 :                     catch (const std::exception &)
     648             :                     {
     649           0 :                         CPLError(CE_Failure, CPLE_OutOfMemory,
     650             :                                  "Out of memory in GetMetadata()");
     651             :                     }
     652             :                 }
     653             :             }
     654           2 :             return m_aosDIMAPMetadata.List();
     655             :         }
     656           8 :         else if (pszDomain && EQUAL(pszDomain, "GEOLOCATION"))
     657             :         {
     658           3 :             return m_aosGEOLOCATION.List();
     659             :         }
     660           5 :         else if (pszDomain && EQUAL(pszDomain, "SUBDATASETS"))
     661             :         {
     662           4 :             if (m_aosSUBDATASETS.empty() && GetGeolocationMetadata())
     663             :             {
     664             :                 m_aosSUBDATASETS.SetNameValue("SUBDATASET_1_NAME",
     665             :                                               CPLSPrintf("%s\"%s\":MAIN",
     666             :                                                          SNAP_TIFF_PREFIX,
     667           1 :                                                          GetDescription()));
     668             :                 m_aosSUBDATASETS.SetNameValue("SUBDATASET_1_DESC",
     669           2 :                                               std::string("Main content of ")
     670           1 :                                                   .append(GetDescription())
     671           2 :                                                   .c_str());
     672             : 
     673             :                 m_aosSUBDATASETS.SetNameValue(
     674             :                     "SUBDATASET_2_NAME",
     675           1 :                     m_aosGEOLOCATION.FetchNameValue("X_DATASET"));
     676             :                 m_aosSUBDATASETS.SetNameValue(
     677           2 :                     "SUBDATASET_2_DESC", std::string("Geolocation array of ")
     678           1 :                                              .append(GetDescription())
     679           2 :                                              .c_str());
     680             :             }
     681           4 :             return m_aosSUBDATASETS.List();
     682             :         }
     683             :     }
     684             : 
     685           2 :     return GDALPamDataset::GetMetadata(pszDomain);
     686             : }
     687             : 
     688             : /************************************************************************/
     689             : /*                         GetMetadataItem()                            */
     690             : /************************************************************************/
     691             : 
     692           4 : const char *SNAPTIFFDataset::GetMetadataItem(const char *pszName,
     693             :                                              const char *pszDomain)
     694             : {
     695           4 :     if (!m_bIsGeolocArray && pszDomain &&
     696           4 :         (EQUAL(pszDomain, "GEOLOCATION") || EQUAL(pszDomain, "SUBDATASETS")))
     697             :     {
     698           3 :         return CSLFetchNameValue(GetMetadata(pszDomain), pszName);
     699             :     }
     700           1 :     return GDALPamDataset::GetMetadataItem(pszName, pszDomain);
     701             : }
     702             : 
     703             : /************************************************************************/
     704             : /*                     GDALRegister_SNAP_TIFF()                         */
     705             : /************************************************************************/
     706             : 
     707        1682 : void GDALRegister_SNAP_TIFF()
     708             : {
     709        1682 :     if (GDALGetDriverByName("SNAP_TIFF") != nullptr)
     710         301 :         return;
     711             : 
     712        1381 :     GDALDriver *poDriver = new GDALDriver();
     713        1381 :     poDriver->SetDescription("SNAP_TIFF");
     714        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
     715        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME,
     716        1381 :                               "Sentinel Application Processing GeoTIFF");
     717        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC,
     718        1381 :                               "drivers/raster/snap_tiff.html");
     719             :     // Declaring the tif extension confuses QGIS
     720             :     // Cf https://github.com/qgis/QGIS/issues/59112
     721             :     // This driver is of too marginal usage to justify causing chaos downstream.
     722             :     // poDriver->SetMetadataItem(GDAL_DMD_MIMETYPE, "image/tiff");
     723             :     // poDriver->SetMetadataItem(GDAL_DMD_EXTENSIONS, "tif tiff");
     724        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
     725             : 
     726        1381 :     poDriver->pfnOpen = SNAPTIFFDataset::Open;
     727        1381 :     poDriver->pfnIdentify = SNAPTIFFDataset::Identify;
     728             : 
     729        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
     730             : }

Generated by: LCOV version 1.14