Coverage Report

Created: 2025-06-22 06:59

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