LCOV - code coverage report
Current view: top level - frmts/mrf - mrf_util.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 180 213 84.5 %
Date: 2025-01-18 12:42:00 Functions: 22 25 88.0 %

          Line data    Source code
       1             : /*
       2             :  * Copyright (c) 2002-2012, California Institute of Technology.
       3             :  * All rights reserved.  Based on Government Sponsored Research under contracts
       4             :  * NAS7-1407 and/or NAS7-03001.
       5             :  *
       6             :  * Redistribution and use in source and binary forms, with or without
       7             :  * modification, are permitted provided that the following conditions are met:
       8             :  *   1. Redistributions of source code must retain the above copyright notice,
       9             :  * this list of conditions and the following disclaimer.
      10             :  *   2. Redistributions in binary form must reproduce the above copyright
      11             :  * notice, this list of conditions and the following disclaimer in the
      12             :  * documentation and/or other materials provided with the distribution.
      13             :  *   3. Neither the name of the California Institute of Technology (Caltech),
      14             :  * its operating division the Jet Propulsion Laboratory (JPL), the National
      15             :  * Aeronautics and Space Administration (NASA), nor the names of its
      16             :  * contributors may be used to endorse or promote products derived from this
      17             :  * software without specific prior written permission.
      18             :  *
      19             :  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
      20             :  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
      21             :  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
      22             :  * ARE DISCLAIMED. IN NO EVENT SHALL THE CALIFORNIA INSTITUTE OF TECHNOLOGY BE
      23             :  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
      24             :  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
      25             :  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
      26             :  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
      27             :  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
      28             :  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
      29             :  * POSSIBILITY OF SUCH DAMAGE.
      30             :  *
      31             :  * Copyright 2014-2021 Esri
      32             :  *
      33             :  * Licensed under the Apache License, Version 2.0 (the "License");
      34             :  * you may not use this file except in compliance with the License.
      35             :  * You may obtain a copy of the License at
      36             :  *
      37             :  * http://www.apache.org/licenses/LICENSE-2.0
      38             :  *
      39             :  * Unless required by applicable law or agreed to in writing, software
      40             :  * distributed under the License is distributed on an "AS IS" BASIS,
      41             :  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
      42             :  * See the License for the specific language governing permissions and
      43             :  * limitations under the License.
      44             :  *
      45             :  *  Functions used by the driver, should have prototypes in the header file
      46             :  *
      47             :  *  Author: Lucian Plesea
      48             :  */
      49             : 
      50             : #include "marfa.h"
      51             : #include <zlib.h>
      52             : #include <algorithm>
      53             : #include <limits>
      54             : #include "mrfdrivercore.h"
      55             : 
      56             : // LERC and QB3 only work on little endian machines
      57             : #if defined(WORDS_BIGENDIAN)
      58             : #undef LERC
      59             : #undef QB3_SUPPORT
      60             : #endif
      61             : 
      62             : CPL_C_START
      63             : void GDALRegister_mrf(void);
      64             : CPL_C_END
      65             : 
      66             : NAMESPACE_MRF_START
      67             : 
      68             : // These have to be positionally in sync with the enums in marfa.h
      69             : static const char *const ILC_N[] = {
      70             : #ifdef HAVE_PNG
      71             :     "PNG",    "PPNG",
      72             : #endif
      73             : #ifdef HAVE_JPEG
      74             :     "JPEG",
      75             : #endif
      76             : #if defined(HAVE_PNG) && defined(HAVE_JPEG)
      77             :     "JPNG",
      78             : #endif
      79             :     "NONE",   "DEFLATE", "TIF",
      80             : #if defined(LERC)
      81             :     "LERC",
      82             : #endif
      83             : #if defined(ZSTD_SUPPORT)
      84             :     "ZSTD",
      85             : #endif
      86             : #if defined(QB3_SUPPORT)
      87             :     "QB3",
      88             : #endif
      89             :     "Unknown"};
      90             : 
      91             : static const char *const ILC_E[] = {
      92             : #ifdef HAVE_PNG
      93             :     ".ppg", ".ppg",
      94             : #endif
      95             : #ifdef HAVE_JPEG
      96             :     ".pjg",
      97             : #endif
      98             : #if defined(HAVE_PNG) && defined(HAVE_JPEG)
      99             :     ".pjp",
     100             : #endif
     101             :     ".til", ".pzp", ".ptf",
     102             : #if defined(LERC)
     103             :     ".lrc",
     104             : #endif
     105             : #if defined(ZSTD_SUPPORT)
     106             :     ".pzs",
     107             : #endif
     108             : #if defined(QB3_SUPPORT)
     109             :     ".pq3",
     110             : #endif
     111             :     ""};
     112             : 
     113             : static const char *const ILO_N[] = {"PIXEL", "BAND", "LINE", "Unknown"};
     114             : 
     115             : char const *const *ILComp_Name = ILC_N;
     116             : char const *const *ILComp_Ext = ILC_E;
     117             : char const *const *ILOrder_Name = ILO_N;
     118             : 
     119             : /**
     120             :  *  Get the string for a compression type
     121             :  */
     122         562 : const char *CompName(ILCompression comp)
     123             : {
     124         562 :     if (comp >= IL_ERR_COMP)
     125           0 :         return ILComp_Name[IL_ERR_COMP];
     126         562 :     return ILComp_Name[comp];
     127             : }
     128             : 
     129             : /**
     130             :  *  Get the string for an order type
     131             :  */
     132         364 : const char *OrderName(ILOrder val)
     133             : {
     134         364 :     if (val >= IL_ERR_ORD)
     135           0 :         return ILOrder_Name[IL_ERR_ORD];
     136         364 :     return ILOrder_Name[val];
     137             : }
     138             : 
     139         459 : ILCompression CompToken(const char *opt, ILCompression def)
     140             : {
     141             :     int i;
     142         459 :     if (nullptr == opt)
     143           0 :         return def;
     144        2418 :     for (i = 0; ILCompression(i) < IL_ERR_COMP; i++)
     145        2418 :         if (EQUAL(opt, ILComp_Name[i]))
     146         459 :             break;
     147         459 :     if (IL_ERR_COMP == ILCompression(i))
     148           0 :         return def;
     149         459 :     return ILCompression(i);
     150             : }
     151             : 
     152             : /**
     153             :  *  Find a compression token
     154             :  */
     155         492 : ILOrder OrderToken(const char *opt, ILOrder def)
     156             : {
     157             :     int i;
     158         492 :     if (nullptr == opt)
     159           0 :         return def;
     160         605 :     for (i = 0; ILOrder(i) < IL_ERR_ORD; i++)
     161         605 :         if (EQUAL(opt, ILOrder_Name[i]))
     162         492 :             break;
     163         492 :     if (IL_ERR_ORD == ILOrder(i))
     164           0 :         return def;
     165         492 :     return ILOrder(i);
     166             : }
     167             : 
     168             : //
     169             : //  Inserters for ILSize and ILIdx types
     170             : //
     171           0 : std::ostream &operator<<(std::ostream &out, const ILSize &sz)
     172             : {
     173           0 :     out << "X=" << sz.x << ",Y=" << sz.y << ",Z=" << sz.z << ",C=" << sz.c
     174           0 :         << ",L=" << sz.l;
     175           0 :     return out;
     176             : }
     177             : 
     178           0 : std::ostream &operator<<(std::ostream &out, const ILIdx &t)
     179             : {
     180           0 :     out << "offset=" << t.offset << ",size=" << t.size;
     181           0 :     return out;
     182             : }
     183             : 
     184             : // Define PPMW to enable this handy debug function
     185             : #ifdef PPMW
     186             : void ppmWrite(const char *fname, const char *data, const ILSize &sz)
     187             : {
     188             :     FILE *fp = fopen(fname, "wb");
     189             :     switch (sz.c)
     190             :     {
     191             :         case 4:  // Strip the alpha
     192             :             fprintf(fp, "P6 %d %d 255\n", sz.x, sz.y);
     193             :             {
     194             :                 char *d = (char *)data;
     195             :                 for (int i = sz.x * sz.y; i; i--)
     196             :                 {
     197             :                     fwrite(d, 3, 1, fp);
     198             :                     d += 4;
     199             :                 }
     200             :             }
     201             :             break;
     202             :         case 3:
     203             :             fprintf(fp, "P6 %d %d 255\n", sz.x, sz.y);
     204             :             fwrite(data, sz.x * sz.y, 3, fp);
     205             :             break;
     206             :         case 1:
     207             :             fprintf(fp, "P5 %d %d 255\n", sz.x, sz.y);
     208             :             fwrite(data, sz.x, sz.y, fp);
     209             :             break;
     210             :         default:
     211             :             fprintf(stderr, "Can't write ppm file with %d bands\n", /*ok*/
     212             :                     sz.c);
     213             :     }
     214             :     fclose(fp);
     215             :     return;
     216             : }
     217             : #endif
     218             : 
     219             : // Returns the size of the index for image and overlays
     220             : // If scale is zero, only base image
     221         310 : GIntBig IdxSize(const ILImage &full, const int scale)
     222             : {
     223         620 :     ILImage img = full;
     224         310 :     img.pagecount = pcount(img.size, img.pagesize);
     225         310 :     GIntBig sz = img.pagecount.l;
     226         349 :     while (scale != 0 && 1 != img.pagecount.x * img.pagecount.y)
     227             :     {
     228          39 :         img.size.x = pcount(img.size.x, scale);
     229          39 :         img.size.y = pcount(img.size.y, scale);
     230          39 :         img.pagecount = pcount(img.size, img.pagesize);
     231          39 :         sz += img.pagecount.l;
     232             :     }
     233             : 
     234         310 :     if (sz >
     235         310 :         std::numeric_limits<GIntBig>::max() / static_cast<int>(sizeof(ILIdx)))
     236             :     {
     237           0 :         CPLError(CE_Failure, CPLE_AppDefined, "IdxSize: integer overflow");
     238           0 :         return 0;
     239             :     }
     240         310 :     return sz * sizeof(ILIdx);
     241             : }
     242             : 
     243         869 : ILImage::ILImage()
     244             :     : dataoffset(0), idxoffset(0), quality(85), pageSizeBytes(0),
     245             :       size(ILSize(1, 1, 1, 1, 0)), pagesize(ILSize(384, 384, 1, 1, 0)),
     246         869 :       pagecount(pcount(size, pagesize)),
     247             : #ifdef HAVE_PNG
     248             :       comp(IL_PNG),
     249             : #else
     250             :       comp(IL_NONE),
     251             : #endif
     252             :       order(IL_Interleaved), nbo(false), hasNoData(FALSE), NoDataValue(0.0),
     253        1738 :       dt(GDT_Unknown), ci(GCI_Undefined)
     254             : {
     255         869 : }
     256             : 
     257             : /**
     258             :  *\brief Get a file name by replacing the extension.
     259             :  * pass the data file name and the default extension starting with .
     260             :  * If name length is not sufficient, it returns the extension
     261             :  * If the input name is curl with parameters, the base file extension gets
     262             :  *changed and parameters are preserved.
     263             :  */
     264             : 
     265        1700 : CPLString getFname(const CPLString &in, const char *ext)
     266             : {
     267        1700 :     if (strlen(in) < strlen(ext))
     268           0 :         return CPLString(ext);
     269             : 
     270        3400 :     CPLString ret(in);
     271             :     // Is it a web file with parameters?
     272        1700 :     size_t extlen = strlen(ext);
     273        1700 :     size_t qmark = ret.find_first_of('?');
     274        1700 :     if (!(qmark != std::string::npos && 0 == in.find("/vsicurl/http") &&
     275             :           qmark >= extlen))
     276        1700 :         qmark = ret.size();
     277        1700 :     return ret.replace(qmark - extlen, extlen, ext);
     278             : }
     279             : 
     280             : /**
     281             :  *\brief Get a file name, either from the configuration or from the default file
     282             :  *name If the token is not defined by CPLGetXMLValue, if the extension of the in
     283             :  *name is .xml, it returns the token with the extension changed to defext.
     284             :  * Otherwise it returns the token itself
     285             :  * It is pretty hard to separate local vs remote due to the gdal file name
     286             :  *ornaments Absolute file names start with: ?:/ or /
     287             :  *
     288             :  */
     289             : 
     290         726 : CPLString getFname(CPLXMLNode *node, const char *token, const CPLString &in,
     291             :                    const char *def)
     292             : {
     293        1452 :     CPLString fn = CPLGetXMLValue(node, token, "");
     294         726 :     if (fn.empty())  // Not provided
     295         714 :         return getFname(in, def);
     296          12 :     size_t slashPos = fn.find_first_of("\\/");
     297             : 
     298             :     // Does it look like an absolute path or we won't find the basename of 'in'
     299          12 :     if (slashPos == 0                       // Starts with slash
     300           6 :         || (slashPos == 2 && fn[1] == ':')  // Starts with disk letter column
     301             :         // Does not start with dots then slash
     302           6 :         || (slashPos != fn.npos && slashPos != fn.find_first_not_of('.')) ||
     303           6 :         EQUALN(in, "<MRF_META>", 10)  // XML string input
     304          18 :         || in.find_first_of("\\/") ==
     305             :                in.npos)  // We can't get a basename from 'in'
     306           6 :         return fn;
     307             : 
     308             :     // Relative path, prepend the path from the in file name
     309          12 :     return in.substr(0, in.find_last_of("\\/") + 1) + fn;
     310             : }
     311             : 
     312             : /**
     313             :  *\brief Extracts a numerical value from a XML node
     314             :  * It works like CPLGetXMLValue except for the default value being
     315             :  * a number instead of a string
     316             :  */
     317             : 
     318        4580 : double getXMLNum(CPLXMLNode *node, const char *pszPath, double def)
     319             : {
     320        4580 :     const char *textval = CPLGetXMLValue(node, pszPath, nullptr);
     321        4580 :     if (textval)
     322        2978 :         return atof(textval);
     323        1602 :     return def;
     324             : }
     325             : 
     326             : //
     327             : // Calculate offset of index, pos is in pages
     328             : //
     329             : 
     330        8986 : GIntBig IdxOffset(const ILSize &pos, const ILImage &img)
     331             : {
     332        8986 :     return img.idxoffset +
     333        8986 :            sizeof(ILIdx) *
     334        8986 :                (pos.c +
     335        8986 :                 img.pagecount.c *
     336        8986 :                     (pos.x + img.pagecount.x *
     337        8986 :                                  (pos.y + img.pagecount.y *
     338        8986 :                                               static_cast<GIntBig>(pos.z))));
     339             : }
     340             : 
     341             : // Is compression type endianness dependent?
     342        8003 : bool is_Endianness_Dependent(GDALDataType dt, ILCompression comp)
     343             : {
     344             :     // Add here all endianness dependent compressions
     345        8003 :     if (IL_ZLIB == comp || IL_NONE == comp)
     346         672 :         if (GDALGetDataTypeSize(dt) > 8)
     347         454 :             return true;
     348        7549 :     return false;
     349             : }
     350             : 
     351         543 : MRFRasterBand *newMRFRasterBand(MRFDataset *pDS, const ILImage &image, int b,
     352             :                                 int level)
     353             : {
     354         543 :     MRFRasterBand *bnd = nullptr;
     355         543 :     CPLErrorReset();
     356         543 :     switch (pDS->current.comp)
     357             :     {
     358             : #ifdef HAVE_PNG
     359         155 :         case IL_PPNG:  // Uses the PNG code, just has a palette in each PNG
     360             :         case IL_PNG:
     361         155 :             bnd = new PNG_Band(pDS, image, b, level);
     362         155 :             break;
     363             : #endif
     364             : #ifdef HAVE_JPEG
     365          49 :         case IL_JPEG:
     366          49 :             bnd = new JPEG_Band(pDS, image, b, level);
     367          49 :             break;
     368             : #endif
     369             : #if defined(HAVE_PNG) && defined(HAVE_JPEG)
     370           0 :         case IL_JPNG:
     371           0 :             bnd = new JPNG_Band(pDS, image, b, level);
     372           0 :             break;
     373             : #endif
     374         137 :         case IL_NONE:
     375         137 :             bnd = new Raw_Band(pDS, image, b, level);
     376         137 :             break;
     377             : #if defined(LERC)
     378          75 :         case IL_LERC:
     379          75 :             bnd = new LERC_Band(pDS, image, b, level);
     380          75 :             break;
     381             : #endif
     382             : #if defined(QB3_SUPPORT)
     383          44 :         case IL_QB3:
     384          44 :             bnd = new QB3_Band(pDS, image, b, level);
     385          44 :             break;
     386             : #endif
     387             :         // ZLIB is just raw + deflate
     388          26 :         case IL_ZLIB:
     389          26 :             bnd = new Raw_Band(pDS, image, b, level);
     390          26 :             bnd->SetDeflate(1);
     391          26 :             break;
     392             :             // Same for ZSTD
     393             : #if defined(ZSTD_SUPPORT)
     394          42 :         case IL_ZSTD:
     395          42 :             bnd = new Raw_Band(pDS, image, b, level);
     396          42 :             bnd->SetZstd(1);
     397          42 :             break;
     398             : #endif
     399          15 :         case IL_TIF:
     400          15 :             if (image.pageSizeBytes > INT_MAX - 1024)
     401           0 :                 return nullptr;
     402          15 :             bnd = new TIF_Band(pDS, image, b, level);
     403          15 :             break;
     404           0 :         default:
     405           0 :             CPLError(CE_Failure, CPLE_AssertionFailed,
     406             :                      "Unsupported MRF compression");
     407           0 :             return nullptr;
     408             :     }
     409             : 
     410             :     // If something was flagged during band creation
     411         543 :     if (CPLGetLastErrorNo() != CE_None)
     412             :     {
     413          53 :         delete bnd;
     414          53 :         return nullptr;
     415             :     }
     416             : 
     417             :     // Copy the RW mode from the dataset
     418         490 :     bnd->SetAccess(pDS->eAccess);
     419         490 :     return bnd;
     420             : }
     421             : 
     422             : /**
     423             :  *\brief log in a given base
     424             :  */
     425          66 : double logbase(double val, double base)
     426             : {
     427          66 :     return log(val) / log(base);
     428             : }
     429             : 
     430             : /**
     431             :  *\brief Is logbase(val, base) an integer?
     432             :  *
     433             :  */
     434          33 : int IsPower(double value, double base)
     435             : {
     436          33 :     double v = logbase(value, base);
     437          33 :     return CPLIsEqual(v, int(v + 0.5));
     438             : }
     439             : 
     440             : /************************************************************************/
     441             : /*                           SearchXMLSiblings()                        */
     442             : /************************************************************************/
     443             : 
     444             : /**
     445             :  *\brief Search for a sibling of the root node with a given name.
     446             :  *
     447             :  * Searches only the next siblings of the node passed in for the named element
     448             :  *or attribute. If the first character of the pszElement is '=', the search
     449             :  *includes the psRoot node
     450             :  *
     451             :  * @param psRoot the root node to search.  This should be a node of type
     452             :  * CXT_Element.  NULL is safe.
     453             :  *
     454             :  * @param pszElement the name of the element or attribute to search for.
     455             :  *
     456             :  * @return The first matching node or NULL on failure.
     457             :  */
     458             : 
     459         256 : CPLXMLNode *SearchXMLSiblings(CPLXMLNode *psRoot, const char *pszElement)
     460             : {
     461         256 :     if (psRoot == nullptr || pszElement == nullptr)
     462           0 :         return nullptr;
     463             : 
     464             :     // If the strings starts with '=', skip it and test the root
     465             :     // If not, start testing with the next sibling
     466         256 :     if (pszElement[0] == '=')
     467           0 :         pszElement++;
     468             :     else
     469         256 :         psRoot = psRoot->psNext;
     470             : 
     471         256 :     for (; psRoot != nullptr; psRoot = psRoot->psNext)
     472         255 :         if ((psRoot->eType == CXT_Element || psRoot->eType == CXT_Attribute) &&
     473         255 :             EQUAL(pszElement, psRoot->pszValue))
     474         255 :             return psRoot;
     475           1 :     return nullptr;
     476             : }
     477             : 
     478             : //
     479             : // Print a double so it can be read with strod while preserving precision
     480             : // Unfortunately this is not quite possible or portable enough at this time
     481             : //
     482        3508 : CPLString PrintDouble(double d, const char *frmt)
     483             : {
     484        7016 :     CPLString res;
     485        3508 :     res.FormatC(d, nullptr);
     486        3508 :     if (CPLStrtod(res.c_str(), nullptr) == d)
     487        3490 :         return res;
     488             : 
     489             :     //  This would be the right code with a C99 compiler that supports %a
     490             :     //  readback in strod()
     491             :     //    return CPLString().Printf("%a",d);
     492             : 
     493          36 :     return CPLString().FormatC(d, frmt);
     494             : }
     495             : 
     496        3340 : void XMLSetAttributeVal(CPLXMLNode *parent, const char *pszName,
     497             :                         const char *pszVal)
     498             : {
     499        3340 :     CPLCreateXMLNode(parent, CXT_Attribute, pszName);
     500        3340 :     CPLSetXMLValue(parent, pszName, pszVal);
     501        3340 : }
     502             : 
     503        3336 : void XMLSetAttributeVal(CPLXMLNode *parent, const char *pszName,
     504             :                         const double val, const char *frmt)
     505             : {
     506        3336 :     XMLSetAttributeVal(parent, pszName, PrintDouble(val, frmt));
     507        3336 : }
     508             : 
     509         676 : CPLXMLNode *XMLSetAttributeVal(CPLXMLNode *parent, const char *pszName,
     510             :                                const ILSize &sz, const char *frmt)
     511             : {
     512         676 :     CPLXMLNode *node = CPLCreateXMLNode(parent, CXT_Element, pszName);
     513         676 :     XMLSetAttributeVal(node, "x", sz.x, frmt);
     514         676 :     XMLSetAttributeVal(node, "y", sz.y, frmt);
     515         676 :     if (sz.z != 1)
     516           0 :         XMLSetAttributeVal(node, "z", sz.z, frmt);
     517         676 :     XMLSetAttributeVal(node, "c", sz.c, frmt);
     518         676 :     return node;
     519             : }
     520             : 
     521             : //
     522             : // Prints a vector of doubles into a string and sets that string as the value of
     523             : // an XML attribute If all values are the same, it only prints one
     524             : //
     525         129 : void XMLSetAttributeVal(CPLXMLNode *parent, const char *pszName,
     526             :                         std::vector<double> const &values)
     527             : {
     528         129 :     if (values.empty())
     529          59 :         return;
     530             : 
     531         140 :     CPLString value;
     532          70 :     double val = values[0];
     533          70 :     int single_val = true;
     534         140 :     for (int i = 0; i < int(values.size()); i++)
     535             :     {
     536          70 :         if (val != values[i])
     537           0 :             single_val = false;
     538          70 :         value.append(PrintDouble(values[i]) + " ");
     539             :     }
     540          70 :     value.pop_back();  // Cut the last space
     541          70 :     if (single_val)
     542          70 :         value = PrintDouble(values[0]);
     543          70 :     CPLCreateXMLNode(parent, CXT_Attribute, pszName);
     544          70 :     CPLSetXMLValue(parent, pszName, value);
     545             : }
     546             : 
     547             : /**
     548             :  *\brief Verify or make a file that big
     549             :  *
     550             :  * @return true if size is OK or if extend succeeded
     551             :  */
     552             : 
     553         187 : int CheckFileSize(const char *fname, GIntBig sz, GDALAccess eAccess)
     554             : {
     555             : 
     556             :     VSIStatBufL statb;
     557         187 :     if (VSIStatL(fname, &statb))
     558           0 :         return false;
     559         187 :     if (statb.st_size >= sz)
     560          84 :         return true;
     561             : 
     562             :     // Don't change anything unless updating
     563         103 :     if (eAccess != GA_Update)
     564           0 :         return false;
     565             : 
     566             :     // There is no ftruncate in VSI, only truncate()
     567         103 :     VSILFILE *ifp = VSIFOpenL(fname, "r+b");
     568         103 :     if (ifp == nullptr)
     569           0 :         return false;
     570             : 
     571         103 :     int ret = VSIFTruncateL(ifp, sz);
     572         103 :     VSIFCloseL(ifp);
     573         103 :     return !ret;
     574             : }
     575             : 
     576             : NAMESPACE_MRF_END
     577             : 
     578             : /************************************************************************/
     579             : /*                          GDALRegister_mrf()                          */
     580             : /************************************************************************/
     581             : 
     582             : USING_NAMESPACE_MRF
     583             : 
     584        1682 : void GDALRegister_mrf()
     585             : {
     586        1682 :     if (GDALGetDriverByName(DRIVER_NAME) != nullptr)
     587         301 :         return;
     588             : 
     589        1381 :     GDALDriver *driver = new GDALDriver();
     590        1381 :     MRFDriverSetCommonMetadata(driver);
     591             : 
     592        1381 :     driver->SetMetadataItem(
     593             :         GDAL_DMD_CREATIONOPTIONLIST,
     594             :         "<CreationOptionList>"
     595             :         "   <Option name='COMPRESS' type='string-select' "
     596             : #ifdef HAVE_PNG
     597             :         "default='PNG' description='PPNG = Palette PNG; DEFLATE = zlib '>"
     598             : #else
     599             :         "default='NONE' description='DEFLATE = zlib '>"
     600             : #endif
     601             : #ifdef HAVE_JPEG
     602             :         "       <Value>JPEG</Value>"
     603             : #endif
     604             : #ifdef HAVE_PNG
     605             :         "       <Value>PNG</Value>"
     606             :         "       <Value>PPNG</Value>"
     607             : #endif
     608             : #if defined(HAVE_JPEG) && defined(HAVE_PNG)
     609             :         "       <Value>JPNG</Value>"
     610             : #endif
     611             :         "       <Value>TIF</Value>"
     612             :         "       <Value>DEFLATE</Value>"
     613             :         "       <Value>NONE</Value>"
     614             : #if defined(LERC)
     615             :         "       <Value>LERC</Value>"
     616             : #endif
     617             : #if defined(ZSTD_SUPPORT)
     618             :         "       <Value>ZSTD</Value>"
     619             : #endif
     620             : #if defined(QB3_SUPPORT)
     621             :         "       <Value>QB3</Value>"
     622             : #endif
     623             :         "   </Option>"
     624             :         "   <Option name='INTERLEAVE' type='string-select' default='PIXEL'>"
     625             :         "       <Value>PIXEL</Value>"
     626             :         "       <Value>BAND</Value>"
     627             :         "   </Option>\n"
     628             :         "   <Option name='ZSIZE' type='int' description='Third dimension size' "
     629             :         "default='1'/>"
     630             :         "   <Option name='QUALITY' type='int' description='Compression "
     631             :         "dependent control value, for JPEG best=99, bad=0, default=85'/>\n"
     632             :         "   <Option name='BLOCKSIZE' type='int' description='Block size, both "
     633             :         "x and y, default 512'/>\n"
     634             :         "   <Option name='BLOCKXSIZE' type='int' description='Block x size, "
     635             :         "default=512'/>\n"
     636             :         "   <Option name='BLOCKYSIZE' type='int' description='Block y size, "
     637             :         "default=512'/>\n"
     638             :         "   <Option name='NETBYTEORDER' type='boolean' "
     639             :         "description='Force endian for certain compress options, default is "
     640             :         "host order'/>\n"
     641             :         "   <Option name='CACHEDSOURCE' type='string' "
     642             :         "description='The source raster, if this is a cache'/>\n"
     643             :         "   <Option name='UNIFORM_SCALE' type='int' description='Scale of "
     644             :         "overlays in MRF, usually 2'/>\n"
     645             :         "   <Option name='NOCOPY' type='boolean' description='Leave created "
     646             :         "MRF empty, default=no'/>\n"
     647             :         "   <Option name='DATANAME' type='string' description='Data file "
     648             :         "name'/>\n"
     649             :         "   <Option name='INDEXNAME' type='string' description='Index file "
     650             :         "name'/>\n"
     651             :         "   <Option name='SPACING' type='int' "
     652             :         "description='Leave this many unused bytes before each tile, "
     653             :         "default=0'/>\n"
     654             :         "   <Option name='PHOTOMETRIC' type='string-select' default='DEFAULT' "
     655             :         "description='Band interpretation, may affect block encoding'>\n"
     656             :         "       <Value>MULTISPECTRAL</Value>"
     657             :         "       <Value>RGB</Value>"
     658             :         "       <Value>YCC</Value>"
     659             :         "   </Option>\n"
     660             :         "   <Option name='OPTIONS' type='string' description='\n"
     661             :         "     Compression dependent parameters, space separated:\n"
     662             : #if defined(ZSTD_SUPPORT)
     663             :         "       ZSTD - boolean, enable libzstd as final stage, preferred over "
     664             :         "DEFLATE\n"
     665             : #endif
     666             :         "       DEFLATE - boolean, enable zlib as final stage\n"
     667             :         "       GZ - boolean, for DEFLATE enable gzip headers instead of zlib "
     668             :         "ones when using zlib\n"
     669             :         "       RAWZ - boolean, for DEFLATE disable all zlib headers\n"
     670             :         "       Z_STRATEGY - Z_HUFFMAN_ONLY | Z_FILTERED | Z_RLE | Z_FIXED: "
     671             :         "restricts DEFLATE and PNG strategy\n"
     672             : #if defined(LERC)
     673             :         "       LERC_PREC - numeric, set LERC precision, defaults to 0.5 for "
     674             :         "int and 0.001 for float\n"
     675             :         "       V1 - boolean, use LERC V1 (older) format\n"
     676             :         "       L2_VER - numeric, encode specific version of Lerc, default is "
     677             :         "library default\n"
     678             :         "                except for single band or INTERLEAVE=BAND, when it "
     679             :         "defaults to 2\n"
     680             : #endif
     681             :         "       OPTIMIZE - boolean, for JPEG, enables Huffman table "
     682             :         "optimization\n"
     683             : #if defined(BRUNSLI)
     684             :         "       JFIF - boolean, for JPEG, disable brunsli encoding\n"
     685             : #endif
     686             :         "'/>"
     687        1381 :         "</CreationOptionList>\n");
     688             : 
     689        1381 :     driver->pfnOpen = MRFDataset::Open;
     690        1381 :     driver->pfnCreateCopy = MRFDataset::CreateCopy;
     691        1381 :     driver->pfnCreate = MRFDataset::Create;
     692        1381 :     driver->pfnDelete = MRFDataset::Delete;
     693        1381 :     GetGDALDriverManager()->RegisterDriver(driver);
     694             : }
     695             : 
     696             : extern "C" void CPL_DLL GDALRegister_MRF(void);
     697             : 
     698           0 : void GDALRegister_MRF()
     699             : {
     700           0 :     GDALRegister_mrf();
     701           0 : }

Generated by: LCOV version 1.14