Coverage Report

Created: 2025-11-16 06:25

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
0
VRTDriver::VRTDriver() : papszSourceParsers(nullptr)
32
0
{
33
#if 0
34
    pDeserializerData = GDALRegisterTransformDeserializer(
35
        "WarpedOverviewTransformer",
36
        VRTWarpedOverviewTransform,
37
        VRTDeserializeWarpedOverviewTransformer );
38
#endif
39
0
}
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
0
{
75
0
    std::lock_guard oLock(m_oMutex);
76
0
    if (pszDomain && EQUAL(pszDomain, "SourceParsers"))
77
0
        return papszSourceParsers;
78
79
0
    return GDALDriver::GetMetadata(pszDomain);
80
0
}
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
0
{
109
0
    m_oMapSourceParser[pszElementName] = pfnParser;
110
111
    // Below won't work on architectures with "capability pointers"
112
113
0
    char szPtrValue[128] = {'\0'};
114
0
    void *ptr;
115
0
    CPL_STATIC_ASSERT(sizeof(pfnParser) == sizeof(void *));
116
0
    memcpy(&ptr, &pfnParser, sizeof(void *));
117
0
    int nRet = CPLPrintPointer(szPtrValue, ptr, sizeof(szPtrValue));
118
0
    szPtrValue[nRet] = 0;
119
120
0
    papszSourceParsers =
121
0
        CSLSetNameValue(papszSourceParsers, pszElementName, szPtrValue);
122
0
}
123
124
/************************************************************************/
125
/*                            ParseSource()                             */
126
/************************************************************************/
127
128
VRTSource *VRTDriver::ParseSource(const CPLXMLNode *psSrc,
129
                                  const char *pszVRTPath,
130
                                  VRTMapSharedResources &oMapSharedSources)
131
132
0
{
133
134
0
    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
0
    if (!m_oMapSourceParser.empty())
142
0
    {
143
0
        auto oIter = m_oMapSourceParser.find(psSrc->pszValue);
144
0
        if (oIter != m_oMapSourceParser.end())
145
0
        {
146
0
            return oIter->second(psSrc, pszVRTPath, oMapSharedSources);
147
0
        }
148
0
        return nullptr;
149
0
    }
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
0
{
179
0
    CPLAssert(nullptr != poSrcDS);
180
181
0
    VRTDataset *poSrcVRTDS = nullptr;
182
183
0
    void *pHandle = poSrcDS->GetInternalHandle("VRT_DATASET");
184
0
    if (pHandle && poSrcDS->GetInternalHandle(nullptr) == nullptr)
185
0
    {
186
0
        poSrcVRTDS = static_cast<VRTDataset *>(pHandle);
187
0
    }
188
0
    else
189
0
    {
190
0
        poSrcVRTDS = dynamic_cast<VRTDataset *>(poSrcDS);
191
0
    }
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
0
    if (poSrcVRTDS)
199
0
    {
200
201
        /* --------------------------------------------------------------------
202
         */
203
        /*      Convert tree to a single block of XML text. */
204
        /* --------------------------------------------------------------------
205
         */
206
0
        char *pszVRTPath = CPLStrdup(CPLGetPathSafe(pszFilename).c_str());
207
0
        poSrcVRTDS->UnsetPreservedRelativeFilenames();
208
0
        CPLXMLNode *psDSTree = poSrcVRTDS->SerializeToXML(pszVRTPath);
209
210
0
        char *pszXML = CPLSerializeXMLTree(psDSTree);
211
212
0
        CPLDestroyXMLNode(psDSTree);
213
214
0
        CPLFree(pszVRTPath);
215
216
        /* --------------------------------------------------------------------
217
         */
218
        /*      Write to disk. */
219
        /* --------------------------------------------------------------------
220
         */
221
0
        GDALDataset *pCopyDS = nullptr;
222
223
0
        if (0 != strlen(pszFilename))
224
0
        {
225
0
            VSILFILE *fpVRT = VSIFOpenL(pszFilename, "wb");
226
0
            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
0
            bool bRet = VSIFWriteL(pszXML, strlen(pszXML), 1, fpVRT) > 0;
235
0
            if (VSIFCloseL(fpVRT) != 0)
236
0
                bRet = false;
237
238
0
            if (bRet)
239
0
                pCopyDS = GDALDataset::Open(
240
0
                    pszFilename,
241
0
                    GDAL_OF_RASTER | GDAL_OF_MULTIDIM_RASTER | GDAL_OF_UPDATE);
242
0
        }
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
0
        CPLFree(pszXML);
252
253
0
        return pCopyDS;
254
0
    }
255
256
    /* -------------------------------------------------------------------- */
257
    /*      Multidimensional raster ?                                       */
258
    /* -------------------------------------------------------------------- */
259
0
    auto poSrcGroup = poSrcDS->GetRootGroup();
260
0
    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
0
    auto poVRTDS = VRTDataset::CreateVRTDataset(
282
0
        pszFilename, poSrcDS->GetRasterXSize(), poSrcDS->GetRasterYSize(), 0,
283
0
        GDT_Byte, papszOptions);
284
0
    if (poVRTDS == nullptr)
285
0
        return nullptr;
286
287
    /* -------------------------------------------------------------------- */
288
    /*      Do we have a geotransform?                                      */
289
    /* -------------------------------------------------------------------- */
290
0
    GDALGeoTransform gt;
291
0
    if (poSrcDS->GetGeoTransform(gt) == CE_None)
292
0
    {
293
0
        poVRTDS->SetGeoTransform(gt);
294
0
    }
295
296
    /* -------------------------------------------------------------------- */
297
    /*      Copy projection                                                 */
298
    /* -------------------------------------------------------------------- */
299
0
    poVRTDS->SetSpatialRef(poSrcDS->GetSpatialRef());
300
301
    /* -------------------------------------------------------------------- */
302
    /*      Emit dataset level metadata.                                    */
303
    /* -------------------------------------------------------------------- */
304
0
    const char *pszCopySrcMDD =
305
0
        CSLFetchNameValueDef(papszOptions, "COPY_SRC_MDD", "AUTO");
306
0
    char **papszSrcMDD = CSLFetchNameValueMultiple(papszOptions, "SRC_MDD");
307
0
    if (EQUAL(pszCopySrcMDD, "AUTO") || CPLTestBool(pszCopySrcMDD) ||
308
0
        papszSrcMDD)
309
0
    {
310
0
        if (!papszSrcMDD || CSLFindString(papszSrcMDD, "") >= 0 ||
311
0
            CSLFindString(papszSrcMDD, "_DEFAULT_") >= 0)
312
0
        {
313
0
            poVRTDS->SetMetadata(poSrcDS->GetMetadata());
314
0
        }
315
316
        /* -------------------------------------------------------------------- */
317
        /*      Copy any special domains that should be transportable.          */
318
        /* -------------------------------------------------------------------- */
319
0
        constexpr const char *apszDefaultDomains[] = {"RPC", "IMD",
320
0
                                                      "GEOLOCATION"};
321
0
        for (const char *pszDomain : apszDefaultDomains)
322
0
        {
323
0
            if (!papszSrcMDD || CSLFindString(papszSrcMDD, pszDomain) >= 0)
324
0
            {
325
0
                char **papszMD = poSrcDS->GetMetadata(pszDomain);
326
0
                if (papszMD)
327
0
                    poVRTDS->SetMetadata(papszMD, pszDomain);
328
0
            }
329
0
        }
330
331
0
        if ((!EQUAL(pszCopySrcMDD, "AUTO") && CPLTestBool(pszCopySrcMDD)) ||
332
0
            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
0
    }
375
0
    CSLDestroy(papszSrcMDD);
376
377
0
    {
378
0
        const char *pszInterleave =
379
0
            poSrcDS->GetMetadataItem("INTERLEAVE", "IMAGE_STRUCTURE");
380
0
        if (pszInterleave)
381
0
        {
382
0
            poVRTDS->SetMetadataItem("INTERLEAVE", pszInterleave,
383
0
                                     "IMAGE_STRUCTURE");
384
0
        }
385
0
    }
386
0
    {
387
0
        const char *pszCompression =
388
0
            poSrcDS->GetMetadataItem("COMPRESSION", "IMAGE_STRUCTURE");
389
0
        if (pszCompression)
390
0
        {
391
0
            poVRTDS->SetMetadataItem("COMPRESSION", pszCompression,
392
0
                                     "IMAGE_STRUCTURE");
393
0
        }
394
0
    }
395
396
    /* -------------------------------------------------------------------- */
397
    /*      GCPs                                                            */
398
    /* -------------------------------------------------------------------- */
399
0
    if (poSrcDS->GetGCPCount() > 0)
400
0
    {
401
0
        poVRTDS->SetGCPs(poSrcDS->GetGCPCount(), poSrcDS->GetGCPs(),
402
0
                         poSrcDS->GetGCPSpatialRef());
403
0
    }
404
405
    /* -------------------------------------------------------------------- */
406
    /*      Loop over all the bands.                                        */
407
    /* -------------------------------------------------------------------- */
408
0
    for (int iBand = 0; iBand < poSrcDS->GetRasterCount(); iBand++)
409
0
    {
410
0
        GDALRasterBand *poSrcBand = poSrcDS->GetRasterBand(iBand + 1);
411
412
        /* --------------------------------------------------------------------
413
         */
414
        /*      Create the band with the appropriate band type. */
415
        /* --------------------------------------------------------------------
416
         */
417
0
        CPLStringList aosAddBandOptions;
418
0
        int nBlockXSize = poVRTDS->GetBlockXSize();
419
0
        int nBlockYSize = poVRTDS->GetBlockYSize();
420
0
        if (!poVRTDS->IsBlockSizeSpecified())
421
0
        {
422
0
            poSrcBand->GetBlockSize(&nBlockXSize, &nBlockYSize);
423
0
        }
424
0
        aosAddBandOptions.SetNameValue("BLOCKXSIZE",
425
0
                                       CPLSPrintf("%d", nBlockXSize));
426
0
        aosAddBandOptions.SetNameValue("BLOCKYSIZE",
427
0
                                       CPLSPrintf("%d", nBlockYSize));
428
0
        poVRTDS->AddBand(poSrcBand->GetRasterDataType(), aosAddBandOptions);
429
430
0
        VRTSourcedRasterBand *poVRTBand = static_cast<VRTSourcedRasterBand *>(
431
0
            poVRTDS->GetRasterBand(iBand + 1));
432
433
        /* --------------------------------------------------------------------
434
         */
435
        /*      Setup source mapping. */
436
        /* --------------------------------------------------------------------
437
         */
438
0
        poVRTBand->AddSimpleSource(poSrcBand);
439
440
        /* --------------------------------------------------------------------
441
         */
442
        /*      Emit various band level metadata. */
443
        /* --------------------------------------------------------------------
444
         */
445
0
        poVRTBand->CopyCommonInfoFrom(poSrcBand);
446
447
0
        const char *pszCompression =
448
0
            poSrcBand->GetMetadataItem("COMPRESSION", "IMAGE_STRUCTURE");
449
0
        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
0
        if ((poSrcBand->GetMaskFlags() &
461
0
             (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
0
    }
470
471
    /* -------------------------------------------------------------------- */
472
    /*      Add dataset mask band                                           */
473
    /* -------------------------------------------------------------------- */
474
0
    if (poSrcDS->GetRasterCount() != 0 &&
475
0
        poSrcDS->GetRasterBand(1) != nullptr &&
476
0
        poSrcDS->GetRasterBand(1)->GetMaskFlags() == GMF_PER_DATASET)
477
0
    {
478
0
        GDALRasterBand *poSrcBand = poSrcDS->GetRasterBand(1);
479
0
        auto poVRTMaskBand = std::make_unique<VRTSourcedRasterBand>(
480
0
            poVRTDS.get(), 0, poSrcBand->GetMaskBand()->GetRasterDataType(),
481
0
            poSrcDS->GetRasterXSize(), poSrcDS->GetRasterYSize());
482
0
        poVRTMaskBand->AddMaskBandSource(poSrcBand);
483
0
        poVRTDS->SetMaskBand(std::move(poVRTMaskBand));
484
0
    }
485
486
0
    if (strcmp(pszFilename, "") != 0)
487
0
    {
488
0
        CPLErrorReset();
489
0
        poVRTDS->FlushCache(true);
490
0
        if (CPLGetLastErrorType() != CE_None)
491
0
        {
492
0
            poVRTDS.reset();
493
0
        }
494
0
    }
495
496
0
    if (pfnProgress)
497
0
        pfnProgress(1.0, "", pProgressData);
498
499
0
    return poVRTDS.release();
500
0
}
501
502
/************************************************************************/
503
/*                          GDALRegister_VRT()                          */
504
/************************************************************************/
505
506
void GDALRegister_VRT()
507
508
0
{
509
0
    auto poDM = GetGDALDriverManager();
510
0
    if (poDM->GetDriverByName("VRT") != nullptr)
511
0
        return;
512
513
0
    static std::once_flag flag;
514
0
    std::call_once(flag,
515
0
                   []()
516
0
                   {
517
                       // First register the pixel functions
518
0
                       GDALRegisterDefaultPixelFunc();
519
520
                       // Register functions for VRTProcessedDataset
521
0
                       GDALVRTRegisterDefaultProcessedDatasetFuncs();
522
0
                   });
523
524
0
    VRTDriver *poDriver = new VRTDriver();
525
526
0
    poDriver->SetDescription("VRT");
527
0
    poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
528
0
    poDriver->SetMetadataItem(GDAL_DCAP_MULTIDIM_RASTER, "YES");
529
0
    poDriver->SetMetadataItem(GDAL_DMD_LONGNAME, "Virtual Raster");
530
0
    poDriver->SetMetadataItem(GDAL_DMD_EXTENSION, "vrt");
531
0
    poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/vrt.html");
532
0
    poDriver->SetMetadataItem(
533
0
        GDAL_DMD_CREATIONDATATYPES,
534
0
        "Byte Int8 Int16 UInt16 Int32 UInt32 Int64 UInt64 "
535
0
        "Float16 Float32 Float64 "
536
0
        "CInt16 CInt32 CFloat16 CFloat32 CFloat64");
537
0
    poDriver->SetMetadataItem(
538
0
        GDAL_DMD_CREATIONOPTIONLIST,
539
0
        "<CreationOptionList>\n"
540
0
        "   <Option name='SUBCLASS' type='string-select' "
541
0
        "default='VRTDataset'>\n"
542
0
        "       <Value>VRTDataset</Value>\n"
543
0
        "       <Value>VRTWarpedDataset</Value>\n"
544
0
        "   </Option>\n"
545
0
        "   <Option name='BLOCKXSIZE' type='int' description='Block width'/>\n"
546
0
        "   <Option name='BLOCKYSIZE' type='int' description='Block height'/>\n"
547
0
        "</CreationOptionList>\n");
548
549
0
    auto poGTiffDrv = poDM->GetDriverByName("GTiff");
550
0
    if (poGTiffDrv)
551
0
    {
552
0
        const char *pszGTiffOvrCO =
553
0
            poGTiffDrv->GetMetadataItem(GDAL_DMD_OVERVIEW_CREATIONOPTIONLIST);
554
0
        if (pszGTiffOvrCO &&
555
0
            STARTS_WITH(pszGTiffOvrCO, "<OverviewCreationOptionList>"))
556
0
        {
557
0
            std::string ocoList =
558
0
                "<OverviewCreationOptionList>"
559
0
                "   <Option name='VIRTUAL' type='boolean' "
560
0
                "default='NO' "
561
0
                "description='Whether virtual overviews rather than "
562
0
                "materialized external GeoTIFF .ovr should be created'/>";
563
0
            ocoList += (pszGTiffOvrCO + strlen("<OverviewCreationOptionList>"));
564
0
            poDriver->SetMetadataItem(GDAL_DMD_OVERVIEW_CREATIONOPTIONLIST,
565
0
                                      ocoList.c_str());
566
0
        }
567
0
    }
568
569
0
    poDriver->SetMetadataItem(
570
0
        GDAL_DMD_MULTIDIM_ARRAY_CREATIONOPTIONLIST,
571
0
        "<MultiDimArrayCreationOptionList>"
572
0
        "   <Option name='BLOCKSIZE' type='int' description='Block size in "
573
0
        "pixels'/>"
574
0
        "</MultiDimArrayCreationOptionList>");
575
576
0
    poDriver->pfnCreateCopy = VRTCreateCopy;
577
0
    poDriver->pfnCreate = VRTDataset::Create;
578
0
    poDriver->pfnCreateMultiDimensional = VRTDataset::CreateMultiDimensional;
579
580
0
#ifndef NO_OPEN
581
0
    poDriver->pfnOpen = VRTDataset::Open;
582
0
    poDriver->pfnIdentify = VRTDataset::Identify;
583
0
    poDriver->pfnDelete = VRTDataset::Delete;
584
585
0
    poDriver->SetMetadataItem(
586
0
        GDAL_DMD_OPENOPTIONLIST,
587
0
        "<OpenOptionList>"
588
0
        "  <Option name='ROOT_PATH' type='string' description='Root path to "
589
0
        "evaluate "
590
0
        "relative paths inside the VRT. Mainly useful for inlined VRT, or "
591
0
        "in-memory "
592
0
        "VRT, where their own directory does not make sense'/>"
593
0
        "<Option name='NUM_THREADS' type='string' description="
594
0
        "'Number of worker threads for reading. Can be set to ALL_CPUS' "
595
0
        "default='ALL_CPUS'/>"
596
0
        "</OpenOptionList>");
597
0
#endif
598
599
0
    poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
600
0
    poDriver->SetMetadataItem(GDAL_DCAP_COORDINATE_EPOCH, "YES");
601
602
0
    poDriver->SetMetadataItem(GDAL_DCAP_UPDATE, "YES");
603
0
    poDriver->SetMetadataItem(GDAL_DMD_UPDATE_ITEMS,
604
0
                              "GeoTransform SRS GCPs NoData "
605
0
                              "ColorInterpretation "
606
0
                              "DatasetMetadata BandMetadata");
607
608
0
    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
0
    poDriver->SetMetadataItem(pszExpressionDialects, "none");
617
0
#endif
618
619
#ifdef GDAL_VRT_ENABLE_MUPARSER
620
    if (gdal::MuParserHasDefineFunUserData())
621
    {
622
        poDriver->SetMetadataItem("MUPARSER_HAS_DEFINE_FUN_USER_DATA", "YES");
623
    }
624
#endif
625
626
0
#ifdef GDAL_VRT_ENABLE_RAWRASTERBAND
627
0
    poDriver->SetMetadataItem("GDAL_VRT_ENABLE_RAWRASTERBAND", "YES");
628
0
#endif
629
630
0
    poDriver->AddSourceParser("SimpleSource", VRTParseCoreSources);
631
0
    poDriver->AddSourceParser("ComplexSource", VRTParseCoreSources);
632
0
    poDriver->AddSourceParser("AveragedSource", VRTParseCoreSources);
633
0
    poDriver->AddSourceParser("NoDataFromMaskSource", VRTParseCoreSources);
634
0
    poDriver->AddSourceParser("KernelFilteredSource", VRTParseFilterSources);
635
0
    poDriver->AddSourceParser("ArraySource", VRTParseArraySource);
636
637
0
    poDM->RegisterDriver(poDriver);
638
0
}
639
640
/*! @endcond */