Coverage Report

Created: 2025-11-15 08:43

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/gdal/frmts/vrt/vrtdriver.cpp
Line
Count
Source
1
/******************************************************************************
2
 *
3
 * Project:  Virtual GDAL Datasets
4
 * Purpose:  Implementation of VRTDriver
5
 * Author:   Frank Warmerdam <warmerdam@pobox.com>
6
 *
7
 ******************************************************************************
8
 * Copyright (c) 2003, Frank Warmerdam <warmerdam@pobox.com>
9
 * Copyright (c) 2009-2013, Even Rouault <even dot rouault at spatialys.com>
10
 *
11
 * SPDX-License-Identifier: MIT
12
 ****************************************************************************/
13
14
#include "vrtdataset.h"
15
16
#include "cpl_minixml.h"
17
#include "cpl_string.h"
18
#include "gdal_alg_priv.h"
19
#include "gdal_frmts.h"
20
#include "gdal_priv.h"
21
#include "vrtexpression.h"
22
23
#include <mutex>
24
25
/*! @cond Doxygen_Suppress */
26
27
/************************************************************************/
28
/*                             VRTDriver()                              */
29
/************************************************************************/
30
31
24
VRTDriver::VRTDriver() : papszSourceParsers(nullptr)
32
24
{
33
#if 0
34
    pDeserializerData = GDALRegisterTransformDeserializer(
35
        "WarpedOverviewTransformer",
36
        VRTWarpedOverviewTransform,
37
        VRTDeserializeWarpedOverviewTransformer );
38
#endif
39
24
}
40
41
/************************************************************************/
42
/*                             ~VRTDriver()                             */
43
/************************************************************************/
44
45
VRTDriver::~VRTDriver()
46
47
0
{
48
0
    CSLDestroy(papszSourceParsers);
49
0
    VRTDerivedRasterBand::Cleanup();
50
#if 0
51
    if(  pDeserializerData )
52
    {
53
        GDALUnregisterTransformDeserializer( pDeserializerData );
54
    }
55
#endif
56
0
}
57
58
/************************************************************************/
59
/*                      GetMetadataDomainList()                         */
60
/************************************************************************/
61
62
char **VRTDriver::GetMetadataDomainList()
63
0
{
64
0
    return BuildMetadataDomainList(GDALDriver::GetMetadataDomainList(), TRUE,
65
0
                                   "SourceParsers", nullptr);
66
0
}
67
68
/************************************************************************/
69
/*                            GetMetadata()                             */
70
/************************************************************************/
71
72
char **VRTDriver::GetMetadata(const char *pszDomain)
73
74
19
{
75
19
    std::lock_guard oLock(m_oMutex);
76
19
    if (pszDomain && EQUAL(pszDomain, "SourceParsers"))
77
0
        return papszSourceParsers;
78
79
19
    return GDALDriver::GetMetadata(pszDomain);
80
19
}
81
82
/************************************************************************/
83
/*                            SetMetadata()                             */
84
/************************************************************************/
85
86
CPLErr VRTDriver::SetMetadata(char **papszMetadata, const char *pszDomain)
87
88
0
{
89
0
    std::lock_guard oLock(m_oMutex);
90
0
    if (pszDomain && EQUAL(pszDomain, "SourceParsers"))
91
0
    {
92
0
        m_oMapSourceParser.clear();
93
0
        CSLDestroy(papszSourceParsers);
94
0
        papszSourceParsers = CSLDuplicate(papszMetadata);
95
0
        return CE_None;
96
0
    }
97
98
0
    return GDALDriver::SetMetadata(papszMetadata, pszDomain);
99
0
}
100
101
/************************************************************************/
102
/*                          AddSourceParser()                           */
103
/************************************************************************/
104
105
void VRTDriver::AddSourceParser(const char *pszElementName,
106
                                VRTSourceParser pfnParser)
107
108
144
{
109
144
    m_oMapSourceParser[pszElementName] = pfnParser;
110
111
    // Below won't work on architectures with "capability pointers"
112
113
144
    char szPtrValue[128] = {'\0'};
114
144
    void *ptr;
115
144
    CPL_STATIC_ASSERT(sizeof(pfnParser) == sizeof(void *));
116
144
    memcpy(&ptr, &pfnParser, sizeof(void *));
117
144
    int nRet = CPLPrintPointer(szPtrValue, ptr, sizeof(szPtrValue));
118
144
    szPtrValue[nRet] = 0;
119
120
144
    papszSourceParsers =
121
144
        CSLSetNameValue(papszSourceParsers, pszElementName, szPtrValue);
122
144
}
123
124
/************************************************************************/
125
/*                            ParseSource()                             */
126
/************************************************************************/
127
128
VRTSource *VRTDriver::ParseSource(const CPLXMLNode *psSrc,
129
                                  const char *pszVRTPath,
130
                                  VRTMapSharedResources &oMapSharedSources)
131
132
61.1k
{
133
134
61.1k
    if (psSrc == nullptr || psSrc->eType != CXT_Element)
135
0
    {
136
0
        CPLError(CE_Failure, CPLE_AppDefined,
137
0
                 "Corrupt or empty VRT source XML document.");
138
0
        return nullptr;
139
0
    }
140
141
61.1k
    if (!m_oMapSourceParser.empty())
142
61.1k
    {
143
61.1k
        auto oIter = m_oMapSourceParser.find(psSrc->pszValue);
144
61.1k
        if (oIter != m_oMapSourceParser.end())
145
29.0k
        {
146
29.0k
            return oIter->second(psSrc, pszVRTPath, oMapSharedSources);
147
29.0k
        }
148
32.0k
        return nullptr;
149
61.1k
    }
150
151
    // Below won't work on architectures with "capability pointers"
152
153
0
    const char *pszParserFunc =
154
0
        CSLFetchNameValue(papszSourceParsers, psSrc->pszValue);
155
0
    if (pszParserFunc == nullptr)
156
0
        return nullptr;
157
158
0
    VRTSourceParser pfnParser;
159
0
    CPL_STATIC_ASSERT(sizeof(pfnParser) == sizeof(void *));
160
0
    void *ptr =
161
0
        CPLScanPointer(pszParserFunc, static_cast<int>(strlen(pszParserFunc)));
162
0
    memcpy(&pfnParser, &ptr, sizeof(void *));
163
164
0
    if (pfnParser == nullptr)
165
0
        return nullptr;
166
167
0
    return pfnParser(psSrc, pszVRTPath, oMapSharedSources);
168
0
}
169
170
/************************************************************************/
171
/*                           VRTCreateCopy()                            */
172
/************************************************************************/
173
174
static GDALDataset *VRTCreateCopy(const char *pszFilename, GDALDataset *poSrcDS,
175
                                  int /* bStrict */, char **papszOptions,
176
                                  GDALProgressFunc pfnProgress,
177
                                  void *pProgressData)
178
51.4k
{
179
51.4k
    CPLAssert(nullptr != poSrcDS);
180
181
51.4k
    VRTDataset *poSrcVRTDS = nullptr;
182
183
51.4k
    void *pHandle = poSrcDS->GetInternalHandle("VRT_DATASET");
184
51.4k
    if (pHandle && poSrcDS->GetInternalHandle(nullptr) == nullptr)
185
0
    {
186
0
        poSrcVRTDS = static_cast<VRTDataset *>(pHandle);
187
0
    }
188
51.4k
    else
189
51.4k
    {
190
51.4k
        poSrcVRTDS = dynamic_cast<VRTDataset *>(poSrcDS);
191
51.4k
    }
192
193
    /* -------------------------------------------------------------------- */
194
    /*      If the source dataset is a virtual dataset then just write      */
195
    /*      it to disk as a special case to avoid extra layers of           */
196
    /*      indirection.                                                    */
197
    /* -------------------------------------------------------------------- */
198
51.4k
    if (poSrcVRTDS)
199
6
    {
200
201
        /* --------------------------------------------------------------------
202
         */
203
        /*      Convert tree to a single block of XML text. */
204
        /* --------------------------------------------------------------------
205
         */
206
6
        char *pszVRTPath = CPLStrdup(CPLGetPathSafe(pszFilename).c_str());
207
6
        poSrcVRTDS->UnsetPreservedRelativeFilenames();
208
6
        CPLXMLNode *psDSTree = poSrcVRTDS->SerializeToXML(pszVRTPath);
209
210
6
        char *pszXML = CPLSerializeXMLTree(psDSTree);
211
212
6
        CPLDestroyXMLNode(psDSTree);
213
214
6
        CPLFree(pszVRTPath);
215
216
        /* --------------------------------------------------------------------
217
         */
218
        /*      Write to disk. */
219
        /* --------------------------------------------------------------------
220
         */
221
6
        GDALDataset *pCopyDS = nullptr;
222
223
6
        if (0 != strlen(pszFilename))
224
6
        {
225
6
            VSILFILE *fpVRT = VSIFOpenL(pszFilename, "wb");
226
6
            if (fpVRT == nullptr)
227
0
            {
228
0
                CPLError(CE_Failure, CPLE_AppDefined, "Cannot create %s",
229
0
                         pszFilename);
230
0
                CPLFree(pszXML);
231
0
                return nullptr;
232
0
            }
233
234
6
            bool bRet = VSIFWriteL(pszXML, strlen(pszXML), 1, fpVRT) > 0;
235
6
            if (VSIFCloseL(fpVRT) != 0)
236
0
                bRet = false;
237
238
6
            if (bRet)
239
6
                pCopyDS = GDALDataset::Open(
240
6
                    pszFilename,
241
6
                    GDAL_OF_RASTER | GDAL_OF_MULTIDIM_RASTER | GDAL_OF_UPDATE);
242
6
        }
243
0
        else
244
0
        {
245
            /* No destination file is given, so pass serialized XML directly. */
246
0
            pCopyDS = GDALDataset::Open(pszXML, GDAL_OF_RASTER |
247
0
                                                    GDAL_OF_MULTIDIM_RASTER |
248
0
                                                    GDAL_OF_UPDATE);
249
0
        }
250
251
6
        CPLFree(pszXML);
252
253
6
        return pCopyDS;
254
6
    }
255
256
    /* -------------------------------------------------------------------- */
257
    /*      Multidimensional raster ?                                       */
258
    /* -------------------------------------------------------------------- */
259
51.4k
    auto poSrcGroup = poSrcDS->GetRootGroup();
260
51.4k
    if (poSrcGroup != nullptr)
261
0
    {
262
0
        auto poDstDS = std::unique_ptr<GDALDataset>(
263
0
            VRTDataset::CreateMultiDimensional(pszFilename, nullptr, nullptr));
264
0
        if (!poDstDS)
265
0
            return nullptr;
266
0
        auto poDstGroup = poDstDS->GetRootGroup();
267
0
        if (!poDstGroup)
268
0
            return nullptr;
269
0
        if (GDALDriver::DefaultCreateCopyMultiDimensional(
270
0
                poSrcDS, poDstDS.get(), false, nullptr, nullptr, nullptr) !=
271
0
            CE_None)
272
0
            return nullptr;
273
0
        if (pfnProgress)
274
0
            pfnProgress(1.0, "", pProgressData);
275
0
        return poDstDS.release();
276
0
    }
277
278
    /* -------------------------------------------------------------------- */
279
    /*      Create the virtual dataset.                                     */
280
    /* -------------------------------------------------------------------- */
281
51.4k
    auto poVRTDS = VRTDataset::CreateVRTDataset(
282
51.4k
        pszFilename, poSrcDS->GetRasterXSize(), poSrcDS->GetRasterYSize(), 0,
283
51.4k
        GDT_Byte, papszOptions);
284
51.4k
    if (poVRTDS == nullptr)
285
0
        return nullptr;
286
287
    /* -------------------------------------------------------------------- */
288
    /*      Do we have a geotransform?                                      */
289
    /* -------------------------------------------------------------------- */
290
51.4k
    GDALGeoTransform gt;
291
51.4k
    if (poSrcDS->GetGeoTransform(gt) == CE_None)
292
4.24k
    {
293
4.24k
        poVRTDS->SetGeoTransform(gt);
294
4.24k
    }
295
296
    /* -------------------------------------------------------------------- */
297
    /*      Copy projection                                                 */
298
    /* -------------------------------------------------------------------- */
299
51.4k
    poVRTDS->SetSpatialRef(poSrcDS->GetSpatialRef());
300
301
    /* -------------------------------------------------------------------- */
302
    /*      Emit dataset level metadata.                                    */
303
    /* -------------------------------------------------------------------- */
304
51.4k
    const char *pszCopySrcMDD =
305
51.4k
        CSLFetchNameValueDef(papszOptions, "COPY_SRC_MDD", "AUTO");
306
51.4k
    char **papszSrcMDD = CSLFetchNameValueMultiple(papszOptions, "SRC_MDD");
307
51.4k
    if (EQUAL(pszCopySrcMDD, "AUTO") || CPLTestBool(pszCopySrcMDD) ||
308
0
        papszSrcMDD)
309
51.4k
    {
310
51.4k
        if (!papszSrcMDD || CSLFindString(papszSrcMDD, "") >= 0 ||
311
0
            CSLFindString(papszSrcMDD, "_DEFAULT_") >= 0)
312
51.4k
        {
313
51.4k
            poVRTDS->SetMetadata(poSrcDS->GetMetadata());
314
51.4k
        }
315
316
        /* -------------------------------------------------------------------- */
317
        /*      Copy any special domains that should be transportable.          */
318
        /* -------------------------------------------------------------------- */
319
51.4k
        constexpr const char *apszDefaultDomains[] = {"RPC", "IMD",
320
51.4k
                                                      "GEOLOCATION"};
321
51.4k
        for (const char *pszDomain : apszDefaultDomains)
322
154k
        {
323
154k
            if (!papszSrcMDD || CSLFindString(papszSrcMDD, pszDomain) >= 0)
324
154k
            {
325
154k
                char **papszMD = poSrcDS->GetMetadata(pszDomain);
326
154k
                if (papszMD)
327
0
                    poVRTDS->SetMetadata(papszMD, pszDomain);
328
154k
            }
329
154k
        }
330
331
51.4k
        if ((!EQUAL(pszCopySrcMDD, "AUTO") && CPLTestBool(pszCopySrcMDD)) ||
332
51.4k
            papszSrcMDD)
333
0
        {
334
0
            char **papszDomainList = poSrcDS->GetMetadataDomainList();
335
0
            constexpr const char *apszReservedDomains[] = {
336
0
                "IMAGE_STRUCTURE", "DERIVED_SUBDATASETS"};
337
0
            for (char **papszIter = papszDomainList; papszIter && *papszIter;
338
0
                 ++papszIter)
339
0
            {
340
0
                const char *pszDomain = *papszIter;
341
0
                if (pszDomain[0] != 0 &&
342
0
                    (!papszSrcMDD ||
343
0
                     CSLFindString(papszSrcMDD, pszDomain) >= 0))
344
0
                {
345
0
                    bool bCanCopy = true;
346
0
                    for (const char *pszOtherDomain : apszDefaultDomains)
347
0
                    {
348
0
                        if (EQUAL(pszDomain, pszOtherDomain))
349
0
                        {
350
0
                            bCanCopy = false;
351
0
                            break;
352
0
                        }
353
0
                    }
354
0
                    if (!papszSrcMDD)
355
0
                    {
356
0
                        for (const char *pszOtherDomain : apszReservedDomains)
357
0
                        {
358
0
                            if (EQUAL(pszDomain, pszOtherDomain))
359
0
                            {
360
0
                                bCanCopy = false;
361
0
                                break;
362
0
                            }
363
0
                        }
364
0
                    }
365
0
                    if (bCanCopy)
366
0
                    {
367
0
                        poVRTDS->SetMetadata(poSrcDS->GetMetadata(pszDomain),
368
0
                                             pszDomain);
369
0
                    }
370
0
                }
371
0
            }
372
0
            CSLDestroy(papszDomainList);
373
0
        }
374
51.4k
    }
375
51.4k
    CSLDestroy(papszSrcMDD);
376
377
51.4k
    {
378
51.4k
        const char *pszInterleave =
379
51.4k
            poSrcDS->GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE");
380
51.4k
        if (pszInterleave)
381
51.4k
        {
382
51.4k
            poVRTDS->SetMetadataItem("INTERLEAVE", pszInterleave,
383
51.4k
                                     "IMAGE_STRUCTURE");
384
51.4k
        }
385
51.4k
    }
386
51.4k
    {
387
51.4k
        const char *pszCompression =
388
51.4k
            poSrcDS->GetMetadataItem("COMPRESSION", "IMAGE_STRUCTURE");
389
51.4k
        if (pszCompression)
390
6.01k
        {
391
6.01k
            poVRTDS->SetMetadataItem("COMPRESSION", pszCompression,
392
6.01k
                                     "IMAGE_STRUCTURE");
393
6.01k
        }
394
51.4k
    }
395
396
    /* -------------------------------------------------------------------- */
397
    /*      GCPs                                                            */
398
    /* -------------------------------------------------------------------- */
399
51.4k
    if (poSrcDS->GetGCPCount() > 0)
400
5.04k
    {
401
5.04k
        poVRTDS->SetGCPs(poSrcDS->GetGCPCount(), poSrcDS->GetGCPs(),
402
5.04k
                         poSrcDS->GetGCPSpatialRef());
403
5.04k
    }
404
405
    /* -------------------------------------------------------------------- */
406
    /*      Loop over all the bands.                                        */
407
    /* -------------------------------------------------------------------- */
408
102k
    for (int iBand = 0; iBand < poSrcDS->GetRasterCount(); iBand++)
409
51.4k
    {
410
51.4k
        GDALRasterBand *poSrcBand = poSrcDS->GetRasterBand(iBand + 1);
411
412
        /* --------------------------------------------------------------------
413
         */
414
        /*      Create the band with the appropriate band type. */
415
        /* --------------------------------------------------------------------
416
         */
417
51.4k
        CPLStringList aosAddBandOptions;
418
51.4k
        int nBlockXSize = poVRTDS->GetBlockXSize();
419
51.4k
        int nBlockYSize = poVRTDS->GetBlockYSize();
420
51.4k
        if (!poVRTDS->IsBlockSizeSpecified())
421
51.4k
        {
422
51.4k
            poSrcBand->GetBlockSize(&nBlockXSize, &nBlockYSize);
423
51.4k
        }
424
51.4k
        aosAddBandOptions.SetNameValue("BLOCKXSIZE",
425
51.4k
                                       CPLSPrintf("%d", nBlockXSize));
426
51.4k
        aosAddBandOptions.SetNameValue("BLOCKYSIZE",
427
51.4k
                                       CPLSPrintf("%d", nBlockYSize));
428
51.4k
        poVRTDS->AddBand(poSrcBand->GetRasterDataType(), aosAddBandOptions);
429
430
51.4k
        VRTSourcedRasterBand *poVRTBand = static_cast<VRTSourcedRasterBand *>(
431
51.4k
            poVRTDS->GetRasterBand(iBand + 1));
432
433
        /* --------------------------------------------------------------------
434
         */
435
        /*      Setup source mapping. */
436
        /* --------------------------------------------------------------------
437
         */
438
51.4k
        poVRTBand->AddSimpleSource(poSrcBand);
439
440
        /* --------------------------------------------------------------------
441
         */
442
        /*      Emit various band level metadata. */
443
        /* --------------------------------------------------------------------
444
         */
445
51.4k
        poVRTBand->CopyCommonInfoFrom(poSrcBand);
446
447
51.4k
        const char *pszCompression =
448
51.4k
            poSrcBand->GetMetadataItem("COMPRESSION", "IMAGE_STRUCTURE");
449
51.4k
        if (pszCompression)
450
0
        {
451
0
            poVRTBand->SetMetadataItem("COMPRESSION", pszCompression,
452
0
                                       "IMAGE_STRUCTURE");
453
0
        }
454
455
        /* --------------------------------------------------------------------
456
         */
457
        /*      Add specific mask band. */
458
        /* --------------------------------------------------------------------
459
         */
460
51.4k
        if ((poSrcBand->GetMaskFlags() &
461
51.4k
             (GMF_PER_DATASET | GMF_ALL_VALID | GMF_NODATA)) == 0)
462
0
        {
463
0
            auto poVRTMaskBand = std::make_unique<VRTSourcedRasterBand>(
464
0
                poVRTDS.get(), 0, poSrcBand->GetMaskBand()->GetRasterDataType(),
465
0
                poSrcDS->GetRasterXSize(), poSrcDS->GetRasterYSize());
466
0
            poVRTMaskBand->AddMaskBandSource(poSrcBand);
467
0
            poVRTBand->SetMaskBand(std::move(poVRTMaskBand));
468
0
        }
469
51.4k
    }
470
471
    /* -------------------------------------------------------------------- */
472
    /*      Add dataset mask band                                           */
473
    /* -------------------------------------------------------------------- */
474
51.4k
    if (poSrcDS->GetRasterCount() != 0 &&
475
51.4k
        poSrcDS->GetRasterBand(1) != nullptr &&
476
51.4k
        poSrcDS->GetRasterBand(1)->GetMaskFlags() == GMF_PER_DATASET)
477
279
    {
478
279
        GDALRasterBand *poSrcBand = poSrcDS->GetRasterBand(1);
479
279
        auto poVRTMaskBand = std::make_unique<VRTSourcedRasterBand>(
480
279
            poVRTDS.get(), 0, poSrcBand->GetMaskBand()->GetRasterDataType(),
481
279
            poSrcDS->GetRasterXSize(), poSrcDS->GetRasterYSize());
482
279
        poVRTMaskBand->AddMaskBandSource(poSrcBand);
483
279
        poVRTDS->SetMaskBand(std::move(poVRTMaskBand));
484
279
    }
485
486
51.4k
    if (strcmp(pszFilename, "") != 0)
487
51.4k
    {
488
51.4k
        CPLErrorReset();
489
51.4k
        poVRTDS->FlushCache(true);
490
51.4k
        if (CPLGetLastErrorType() != CE_None)
491
10
        {
492
10
            poVRTDS.reset();
493
10
        }
494
51.4k
    }
495
496
51.4k
    if (pfnProgress)
497
51.4k
        pfnProgress(1.0, "", pProgressData);
498
499
51.4k
    return poVRTDS.release();
500
51.4k
}
501
502
/************************************************************************/
503
/*                          GDALRegister_VRT()                          */
504
/************************************************************************/
505
506
void GDALRegister_VRT()
507
508
75.4k
{
509
75.4k
    auto poDM = GetGDALDriverManager();
510
75.4k
    if (poDM->GetDriverByName("VRT") != nullptr)
511
75.3k
        return;
512
513
24
    static std::once_flag flag;
514
24
    std::call_once(flag,
515
24
                   []()
516
24
                   {
517
                       // First register the pixel functions
518
24
                       GDALRegisterDefaultPixelFunc();
519
520
                       // Register functions for VRTProcessedDataset
521
24
                       GDALVRTRegisterDefaultProcessedDatasetFuncs();
522
24
                   });
523
524
24
    VRTDriver *poDriver = new VRTDriver();
525
526
24
    poDriver->SetDescription("VRT");
527
24
    poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
528
24
    poDriver->SetMetadataItem(GDAL_DCAP_MULTIDIM_RASTER, "YES");
529
24
    poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "Virtual Raster");
530
24
    poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "vrt");
531
24
    poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/vrt.html");
532
24
    poDriver->SetMetadataItem(
533
24
        GDAL_DMD_CREATIONDATATYPES,
534
24
        "Byte Int8 Int16 UInt16 Int32 UInt32 Int64 UInt64 "
535
24
        "Float16 Float32 Float64 "
536
24
        "CInt16 CInt32 CFloat16 CFloat32 CFloat64");
537
24
    poDriver->SetMetadataItem(
538
24
        GDAL_DMD_CREATIONOPTIONLIST,
539
24
        "<CreationOptionList>\n"
540
24
        "   <Option name='SUBCLASS' type='string-select' "
541
24
        "default='VRTDataset'>\n"
542
24
        "       <Value>VRTDataset</Value>\n"
543
24
        "       <Value>VRTWarpedDataset</Value>\n"
544
24
        "   </Option>\n"
545
24
        "   <Option name='BLOCKXSIZE' type='int' description='Block width'/>\n"
546
24
        "   <Option name='BLOCKYSIZE' type='int' description='Block height'/>\n"
547
24
        "</CreationOptionList>\n");
548
549
24
    auto poGTiffDrv = poDM->GetDriverByName("GTiff");
550
24
    if (poGTiffDrv)
551
24
    {
552
24
        const char *pszGTiffOvrCO =
553
24
            poGTiffDrv->GetMetadataItem(GDAL_DMD_OVERVIEW_CREATIONOPTIONLIST);
554
24
        if (pszGTiffOvrCO &&
555
24
            STARTS_WITH(pszGTiffOvrCO, "<OverviewCreationOptionList>"))
556
24
        {
557
24
            std::string ocoList =
558
24
                "<OverviewCreationOptionList>"
559
24
                "   <Option name='VIRTUAL' type='boolean' "
560
24
                "default='NO' "
561
24
                "description='Whether virtual overviews rather than "
562
24
                "materialized external GeoTIFF .ovr should be created'/>";
563
24
            ocoList += (pszGTiffOvrCO + strlen("<OverviewCreationOptionList>"));
564
24
            poDriver->SetMetadataItem(GDAL_DMD_OVERVIEW_CREATIONOPTIONLIST,
565
24
                                      ocoList.c_str());
566
24
        }
567
24
    }
568
569
24
    poDriver->SetMetadataItem(
570
24
        GDAL_DMD_MULTIDIM_ARRAY_CREATIONOPTIONLIST,
571
24
        "<MultiDimArrayCreationOptionList>"
572
24
        "   <Option name='BLOCKSIZE' type='int' description='Block size in "
573
24
        "pixels'/>"
574
24
        "</MultiDimArrayCreationOptionList>");
575
576
24
    poDriver->pfnCreateCopy = VRTCreateCopy;
577
24
    poDriver->pfnCreate = VRTDataset::Create;
578
24
    poDriver->pfnCreateMultiDimensional = VRTDataset::CreateMultiDimensional;
579
580
24
#ifndef NO_OPEN
581
24
    poDriver->pfnOpen = VRTDataset::Open;
582
24
    poDriver->pfnIdentify = VRTDataset::Identify;
583
24
    poDriver->pfnDelete = VRTDataset::Delete;
584
585
24
    poDriver->SetMetadataItem(
586
24
        GDAL_DMD_OPENOPTIONLIST,
587
24
        "<OpenOptionList>"
588
24
        "  <Option name='ROOT_PATH' type='string' description='Root path to "
589
24
        "evaluate "
590
24
        "relative paths inside the VRT. Mainly useful for inlined VRT, or "
591
24
        "in-memory "
592
24
        "VRT, where their own directory does not make sense'/>"
593
24
        "<Option name='NUM_THREADS' type='string' description="
594
24
        "'Number of worker threads for reading. Can be set to ALL_CPUS' "
595
24
        "default='ALL_CPUS'/>"
596
24
        "</OpenOptionList>");
597
24
#endif
598
599
24
    poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
600
24
    poDriver->SetMetadataItem(GDAL_DCAP_COORDINATE_EPOCH, "YES");
601
602
24
    poDriver->SetMetadataItem(GDAL_DCAP_UPDATE, "YES");
603
24
    poDriver->SetMetadataItem(GDAL_DMD_UPDATE_ITEMS,
604
24
                              "GeoTransform SRS GCPs NoData "
605
24
                              "ColorInterpretation "
606
24
                              "DatasetMetadata BandMetadata");
607
608
24
    const char *pszExpressionDialects = "ExpressionDialects";
609
#if defined(GDAL_VRT_ENABLE_MUPARSER) && defined(GDAL_VRT_ENABLE_EXPRTK)
610
    poDriver->SetMetadataItem(pszExpressionDialects, "muparser,exprtk");
611
#elif defined(GDAL_VRT_ENABLE_MUPARSER)
612
    poDriver->SetMetadataItem(pszExpressionDialects, "muparser");
613
#elif defined(GDAL_VRT_ENABLE_EXPRTK)
614
    poDriver->SetMetadataItem(pszExpressionDialects, "exprtk");
615
#else
616
    poDriver->SetMetadataItem(pszExpressionDialects, "none");
617
#endif
618
619
24
#ifdef GDAL_VRT_ENABLE_MUPARSER
620
24
    if (gdal::MuParserHasDefineFunUserData())
621
24
    {
622
24
        poDriver->SetMetadataItem("MUPARSER_HAS_DEFINE_FUN_USER_DATA", "YES");
623
24
    }
624
24
#endif
625
626
24
#ifdef GDAL_VRT_ENABLE_RAWRASTERBAND
627
24
    poDriver->SetMetadataItem("GDAL_VRT_ENABLE_RAWRASTERBAND", "YES");
628
24
#endif
629
630
24
    poDriver->AddSourceParser("SimpleSource", VRTParseCoreSources);
631
24
    poDriver->AddSourceParser("ComplexSource", VRTParseCoreSources);
632
24
    poDriver->AddSourceParser("AveragedSource", VRTParseCoreSources);
633
24
    poDriver->AddSourceParser("NoDataFromMaskSource", VRTParseCoreSources);
634
24
    poDriver->AddSourceParser("KernelFilteredSource", VRTParseFilterSources);
635
24
    poDriver->AddSourceParser("ArraySource", VRTParseArraySource);
636
637
24
    poDM->RegisterDriver(poDriver);
638
24
}
639
640
/*! @endcond */