/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 */ |