Line data Source code
1 : /******************************************************************************
2 : * Project: Selafin importer
3 : * Purpose: Implementation of OGRSelafinDataSource class.
4 : * Author: François Hissel, francois.hissel@gmail.com
5 : *
6 : ******************************************************************************
7 : * Copyright (c) 2014, François Hissel <francois.hissel@gmail.com>
8 : *
9 : * SPDX-License-Identifier: MIT
10 : ****************************************************************************/
11 :
12 : #include "ogr_selafin.h"
13 : #include "cpl_conv.h"
14 : #include "cpl_string.h"
15 : #include "cpl_vsi_virtual.h"
16 : #include "cpl_vsi.h"
17 : #include "io_selafin.h"
18 :
19 : #include <algorithm>
20 : #include <ctime>
21 :
22 : /************************************************************************/
23 : /* Range */
24 : /************************************************************************/
25 43622 : Range::~Range()
26 : {
27 21811 : deleteList(poVals);
28 21811 : deleteList(poActual);
29 21811 : }
30 :
31 43631 : void Range::deleteList(Range::List *poList)
32 : {
33 43631 : if (poList == nullptr)
34 43631 : return;
35 0 : Range::List *pol = poList;
36 0 : while (pol != nullptr)
37 : {
38 0 : poList = poList->poNext;
39 0 : delete pol;
40 0 : pol = poList;
41 : }
42 : }
43 :
44 3 : void Range::setRange(const char *pszStr)
45 : {
46 3 : deleteList(poVals);
47 3 : deleteList(poActual);
48 3 : poVals = nullptr;
49 3 : Range::List *poEnd = nullptr;
50 3 : if (pszStr == nullptr || pszStr[0] != '[')
51 : {
52 0 : CPLError(CE_Warning, CPLE_IllegalArg, "Invalid range specified\n");
53 3 : return;
54 : }
55 3 : const char *pszc = pszStr;
56 3 : char *psze = nullptr;
57 : SelafinTypeDef eType;
58 3 : while (*pszc != 0 && *pszc != ']')
59 : {
60 3 : pszc++;
61 3 : if (*pszc == 'p' || *pszc == 'P')
62 : {
63 0 : eType = POINTS;
64 0 : pszc++;
65 : }
66 3 : else if (*pszc == 'e' || *pszc == 'E')
67 : {
68 0 : eType = ELEMENTS;
69 0 : pszc++;
70 : }
71 : else
72 3 : eType = ALL;
73 :
74 3 : int nMin = 0;
75 3 : if (*pszc != ':')
76 : {
77 3 : nMin = (int)strtol(pszc, &psze, 10);
78 3 : if (*psze != ':' && *psze != ',' && *psze != ']')
79 : {
80 3 : CPLError(CE_Warning, CPLE_IllegalArg,
81 : "Invalid range specified\n");
82 3 : deleteList(poVals);
83 3 : poVals = nullptr;
84 3 : return;
85 : }
86 0 : pszc = psze;
87 : }
88 0 : int nMax = -1;
89 0 : if (*pszc == ':')
90 : {
91 0 : ++pszc;
92 0 : if (*pszc != ',' && *pszc != ']')
93 : {
94 0 : nMax = (int)strtol(pszc, &psze, 10);
95 0 : if (*psze != ',' && *psze != ']')
96 : {
97 0 : CPLError(CE_Warning, CPLE_IllegalArg,
98 : "Invalid range specified\n");
99 0 : deleteList(poVals);
100 0 : poVals = nullptr;
101 0 : return;
102 : }
103 0 : pszc = psze;
104 : }
105 : }
106 : else
107 0 : nMax = nMin;
108 0 : Range::List *poNew = nullptr;
109 0 : if (eType != ALL)
110 0 : poNew = new Range::List(eType, nMin, nMax, nullptr);
111 : else
112 0 : poNew =
113 : new Range::List(POINTS, nMin, nMax,
114 0 : new Range::List(ELEMENTS, nMin, nMax, nullptr));
115 0 : if (poVals == nullptr)
116 : {
117 0 : poVals = poNew;
118 0 : poEnd = poNew;
119 : }
120 : else
121 : {
122 0 : poEnd->poNext = poNew;
123 0 : poEnd = poNew;
124 : }
125 0 : if (poEnd->poNext != nullptr)
126 0 : poEnd = poEnd->poNext;
127 : }
128 0 : if (*pszc != ']')
129 : {
130 0 : CPLError(CE_Warning, CPLE_IllegalArg, "Invalid range specified\n");
131 0 : deleteList(poVals);
132 0 : poVals = nullptr;
133 : }
134 : }
135 :
136 8 : bool Range::contains(SelafinTypeDef eType, int nValue) const
137 : {
138 8 : if (poVals == nullptr)
139 8 : return true;
140 0 : Range::List *poCur = poActual;
141 0 : while (poCur != nullptr)
142 : {
143 0 : if (poCur->eType == eType && nValue >= poCur->nMin &&
144 0 : nValue <= poCur->nMax)
145 0 : return true;
146 0 : poCur = poCur->poNext;
147 : }
148 0 : return false;
149 : }
150 :
151 0 : void Range::sortList(Range::List *&poList, Range::List *poEnd)
152 : {
153 0 : if (poList == nullptr || poList == poEnd)
154 0 : return;
155 0 : Range::List *pol = poList;
156 0 : Range::List *poBefore = nullptr;
157 0 : Range::List *poBeforeEnd = nullptr;
158 : // poList plays the role of the pivot value. Values greater and smaller are
159 : // sorted on each side of it. The order relation here is POINTS ranges
160 : // first, then sorted by nMin value.
161 0 : while (pol->poNext != poEnd)
162 : {
163 0 : if ((pol->eType == ELEMENTS &&
164 0 : (pol->poNext->eType == POINTS || pol->poNext->nMin < pol->nMin)) ||
165 0 : (pol->eType == POINTS && pol->poNext->eType == POINTS &&
166 0 : pol->poNext->nMin < pol->nMin))
167 : {
168 0 : if (poBefore == nullptr)
169 : {
170 0 : poBefore = pol->poNext;
171 0 : poBeforeEnd = poBefore;
172 : }
173 : else
174 : {
175 0 : poBeforeEnd->poNext = pol->poNext;
176 0 : poBeforeEnd = poBeforeEnd->poNext;
177 : }
178 0 : pol->poNext = pol->poNext->poNext;
179 : }
180 : else
181 0 : pol = pol->poNext;
182 : }
183 0 : if (poBefore != nullptr)
184 0 : poBeforeEnd->poNext = poList;
185 : // Now, poList is well placed. We do the same for the sublists before and
186 : // after poList
187 0 : Range::sortList(poBefore, poList);
188 0 : Range::sortList(poList->poNext, poEnd);
189 : // Finally, we restore the right starting point of the list
190 0 : if (poBefore != nullptr)
191 0 : poList = poBefore;
192 : }
193 :
194 21 : void Range::setMaxValue(int nMaxValueP)
195 : {
196 21 : nMaxValue = nMaxValueP;
197 21 : if (poVals == nullptr)
198 21 : return;
199 : // We keep an internal private copy of the list where the range is
200 : // "resolved", that is simplified to a union of disjoint intervals
201 0 : deleteList(poActual);
202 0 : poActual = nullptr;
203 0 : Range::List *pol = poVals;
204 0 : Range::List *poActualEnd = nullptr;
205 : int nMinT, nMaxT;
206 0 : while (pol != nullptr)
207 : {
208 0 : if (pol->nMin < 0)
209 0 : nMinT = pol->nMin + nMaxValue;
210 : else
211 0 : nMinT = pol->nMin;
212 0 : if (pol->nMin < 0)
213 0 : pol->nMin = 0;
214 0 : if (pol->nMin >= nMaxValue)
215 0 : pol->nMin = nMaxValue - 1;
216 0 : if (pol->nMax < 0)
217 0 : nMaxT = pol->nMax + nMaxValue;
218 : else
219 0 : nMaxT = pol->nMax;
220 0 : if (pol->nMax < 0)
221 0 : pol->nMax = 0;
222 0 : if (pol->nMax >= nMaxValue)
223 0 : pol->nMax = nMaxValue - 1;
224 0 : if (nMaxT < nMinT)
225 0 : continue;
226 0 : if (poActual == nullptr)
227 : {
228 0 : poActual = new Range::List(pol->eType, nMinT, nMaxT, nullptr);
229 0 : poActualEnd = poActual;
230 : }
231 : else
232 : {
233 0 : poActualEnd->poNext =
234 0 : new Range::List(pol->eType, nMinT, nMaxT, nullptr);
235 0 : poActualEnd = poActualEnd->poNext;
236 : }
237 0 : pol = pol->poNext;
238 : }
239 0 : sortList(poActual);
240 : // Now we merge successive ranges when they intersect or are consecutive
241 0 : if (poActual != nullptr)
242 : {
243 0 : pol = poActual;
244 0 : while (pol->poNext != nullptr)
245 : {
246 0 : if (pol->poNext->eType == pol->eType &&
247 0 : pol->poNext->nMin <= pol->nMax + 1)
248 : {
249 0 : if (pol->poNext->nMax > pol->nMax)
250 0 : pol->nMax = pol->poNext->nMax;
251 0 : poActualEnd = pol->poNext->poNext;
252 0 : delete pol->poNext;
253 0 : pol->poNext = poActualEnd;
254 : }
255 : else
256 0 : pol = pol->poNext;
257 : }
258 : }
259 : }
260 :
261 21 : size_t Range::getSize() const
262 : {
263 21 : if (poVals == nullptr)
264 21 : return nMaxValue * 2;
265 0 : Range::List *pol = poActual;
266 0 : size_t nSize = 0;
267 0 : while (pol != nullptr)
268 : {
269 0 : nSize += (pol->nMax - pol->nMin + 1);
270 0 : pol = pol->poNext;
271 : }
272 0 : return nSize;
273 : }
274 :
275 : /************************************************************************/
276 : /* OGRSelafinDataSource() */
277 : /************************************************************************/
278 :
279 21811 : OGRSelafinDataSource::OGRSelafinDataSource()
280 : : pszName(nullptr), papoLayers(nullptr), nLayers(0), bUpdate(false),
281 21811 : poHeader(nullptr), poSpatialRef(nullptr)
282 : {
283 21811 : }
284 :
285 : /************************************************************************/
286 : /* ~OGRSelafinDataSource() */
287 : /************************************************************************/
288 :
289 43622 : OGRSelafinDataSource::~OGRSelafinDataSource()
290 : {
291 21825 : for (int i = 0; i < nLayers; i++)
292 14 : delete papoLayers[i];
293 21811 : CPLFree(papoLayers);
294 21811 : CPLFree(pszName);
295 21811 : delete poHeader;
296 21811 : if (poSpatialRef != nullptr)
297 4 : poSpatialRef->Release();
298 43622 : }
299 :
300 : /************************************************************************/
301 : /* TestCapability() */
302 : /************************************************************************/
303 :
304 18 : int OGRSelafinDataSource::TestCapability(const char *pszCap)
305 : {
306 18 : if (EQUAL(pszCap, ODsCCreateLayer))
307 17 : return TRUE;
308 1 : else if (EQUAL(pszCap, ODsCDeleteLayer))
309 1 : return TRUE;
310 0 : else if (EQUAL(pszCap, ODsCCreateGeomFieldAfterCreateLayer))
311 0 : return FALSE;
312 : else
313 0 : return FALSE;
314 : }
315 :
316 : /************************************************************************/
317 : /* GetLayer() */
318 : /************************************************************************/
319 :
320 20 : OGRLayer *OGRSelafinDataSource::GetLayer(int iLayer)
321 : {
322 20 : if (iLayer < 0 || iLayer >= nLayers)
323 0 : return nullptr;
324 : else
325 20 : return papoLayers[iLayer];
326 : }
327 :
328 : /************************************************************************/
329 : /* Open() */
330 : /************************************************************************/
331 21811 : int OGRSelafinDataSource::Open(const char *pszFilename, int bUpdateIn,
332 : int bCreate)
333 : {
334 : // Check if a range is set and extract it and the filename.
335 21811 : const char *pszc = pszFilename;
336 21811 : if (*pszFilename == 0)
337 439 : return FALSE;
338 931433 : while (*pszc)
339 910061 : ++pszc;
340 21372 : if (*(pszc - 1) == ']')
341 : {
342 3 : --pszc;
343 37 : while (pszc != pszFilename && *pszc != '[')
344 34 : pszc--;
345 3 : if (pszc == pszFilename)
346 0 : return FALSE;
347 3 : poRange.setRange(pszc);
348 : }
349 21372 : pszName = CPLStrdup(pszFilename);
350 21372 : pszName[pszc - pszFilename] = 0;
351 21372 : bUpdate = CPL_TO_BOOL(bUpdateIn);
352 21372 : if (bCreate && EQUAL(pszName, "/vsistdout/"))
353 0 : return TRUE;
354 : /* For writable /vsizip/, do nothing more */
355 21372 : if (bCreate && STARTS_WITH(pszName, "/vsizip/"))
356 0 : return TRUE;
357 42744 : CPLString osFilename(pszName);
358 : // Determine what sort of object this is.
359 : VSIStatBufL sStatBuf;
360 21372 : if (VSIStatExL(osFilename, &sStatBuf, VSI_STAT_NATURE_FLAG) != 0)
361 21351 : return FALSE;
362 :
363 : // Is this a single Selafin file?
364 21 : if (VSI_ISREG(sStatBuf.st_mode))
365 21 : return OpenTable(pszName);
366 :
367 : // Is this a single a ZIP file with only a Selafin file inside ?
368 0 : if (STARTS_WITH(osFilename, "/vsizip/") && VSI_ISREG(sStatBuf.st_mode))
369 : {
370 0 : char **papszFiles = VSIReadDir(osFilename);
371 0 : if (CSLCount(papszFiles) != 1)
372 : {
373 0 : CSLDestroy(papszFiles);
374 0 : return FALSE;
375 : }
376 0 : osFilename = CPLFormFilenameSafe(osFilename, papszFiles[0], nullptr);
377 0 : CSLDestroy(papszFiles);
378 0 : return OpenTable(osFilename);
379 : }
380 :
381 : #ifdef notdef
382 : // Otherwise it has to be a directory.
383 : if (!VSI_ISDIR(sStatBuf.st_mode))
384 : return FALSE;
385 :
386 : // Scan through for entries which look like Selafin files
387 : int nNotSelafinCount = 0, i;
388 : char **papszNames = VSIReadDir(osFilename);
389 : for (i = 0; papszNames != NULL && papszNames[i] != NULL; i++)
390 : {
391 : const CPLString oSubFilename =
392 : CPLFormFilenameSafe(osFilename, papszNames[i], NULL);
393 : if (EQUAL(papszNames[i], ".") || EQUAL(papszNames[i], ".."))
394 : continue;
395 : if (VSIStatL(oSubFilename, &sStatBuf) != 0 ||
396 : !VSI_ISREG(sStatBuf.st_mode))
397 : {
398 : nNotSelafinCount++;
399 : continue;
400 : }
401 : if (!OpenTable(oSubFilename))
402 : {
403 : CPLDebug("Selafin", "Cannot open %s", oSubFilename.c_str());
404 : nNotSelafinCount++;
405 : continue;
406 : }
407 : }
408 : CSLDestroy(papszNames);
409 :
410 : // We presume that this is indeed intended to be a Selafin datasource if
411 : // over half the files were Selafin files.
412 : return nNotSelafinCount < nLayers;
413 : #else
414 0 : return FALSE;
415 : #endif
416 : }
417 :
418 : /************************************************************************/
419 : /* OpenTable() */
420 : /************************************************************************/
421 21 : int OGRSelafinDataSource::OpenTable(const char *pszFilename)
422 : {
423 : #ifdef DEBUG_VERBOSE
424 : CPLDebug("Selafin", "OpenTable(%s,%i)", pszFilename,
425 : static_cast<int>(bUpdate));
426 : #endif
427 : // Open the file
428 21 : VSILFILE *fp = nullptr;
429 21 : if (bUpdate)
430 : {
431 20 : fp = VSIFOpenExL(pszFilename, "rb+", true);
432 : }
433 : else
434 : {
435 1 : fp = VSIFOpenExL(pszFilename, "rb", true);
436 : }
437 :
438 21 : if (fp == nullptr)
439 : {
440 0 : CPLError(CE_Warning, CPLE_OpenFailed, "Failed to open %s.",
441 : VSIGetLastErrorMsg());
442 0 : return FALSE;
443 : }
444 21 : if (!bUpdate && strstr(pszFilename, "/vsigzip/") == nullptr &&
445 1 : strstr(pszFilename, "/vsizip/") == nullptr)
446 1 : fp = (VSILFILE *)VSICreateBufferedReaderHandle((VSIVirtualHandle *)fp);
447 :
448 : // Quickly check if the file is in Selafin format, before actually starting
449 : // to read to make it faster
450 : char szBuf[9];
451 21 : VSIFReadL(szBuf, 1, 4, fp);
452 21 : if (szBuf[0] != 0 || szBuf[1] != 0 || szBuf[2] != 0 || szBuf[3] != 0x50)
453 : {
454 0 : VSIFCloseL(fp);
455 0 : return FALSE;
456 : }
457 21 : VSIFSeekL(fp, 84, SEEK_SET);
458 21 : VSIFReadL(szBuf, 1, 8, fp);
459 21 : if (szBuf[0] != 0 || szBuf[1] != 0 || szBuf[2] != 0 || szBuf[3] != 0x50 ||
460 21 : szBuf[4] != 0 || szBuf[5] != 0 || szBuf[6] != 0 || szBuf[7] != 8)
461 : {
462 0 : VSIFCloseL(fp);
463 0 : return FALSE;
464 : }
465 : /* VSIFSeekL(fp,76,SEEK_SET);
466 : VSIFReadL(szBuf,1,8,fp);
467 : if (STRNCASECMP(szBuf,"Seraphin",8)!=0 && STRNCASECMP(szBuf,"Serafin",7)!=0)
468 : { VSIFCloseL(fp); return FALSE;
469 : } */
470 :
471 : // Get layer base name
472 42 : CPLString osBaseLayerName = CPLGetBasenameSafe(pszFilename);
473 :
474 : // Read header of file to get common information for all layers
475 : // poHeader now owns fp
476 21 : poHeader = Selafin::read_header(fp, pszFilename);
477 21 : if (poHeader == nullptr)
478 : {
479 0 : CPLError(CE_Failure, CPLE_OpenFailed,
480 : "Failed to open %s, wrong format.\n", pszFilename);
481 0 : return FALSE;
482 : }
483 21 : if (poHeader->nEpsg != 0)
484 : {
485 2 : poSpatialRef = new OGRSpatialReference();
486 2 : poSpatialRef->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
487 2 : if (poSpatialRef->importFromEPSG(poHeader->nEpsg) != OGRERR_NONE)
488 : {
489 0 : CPLError(CE_Warning, CPLE_AppDefined,
490 : "EPSG %d not found. Could not set datasource SRS.\n",
491 0 : poHeader->nEpsg);
492 0 : delete poSpatialRef;
493 0 : poSpatialRef = nullptr;
494 : }
495 : }
496 :
497 : // To prevent int overflow in poRange.getSize() call where we do
498 : // nSteps * 2
499 21 : if (poHeader->nSteps >= INT_MAX / 2)
500 : {
501 0 : CPLError(CE_Failure, CPLE_OpenFailed, "Invalid nSteps value");
502 0 : return FALSE;
503 : }
504 :
505 : // Create two layers for each selected time step: one for points, the other
506 : // for elements
507 21 : poRange.setMaxValue(poHeader->nSteps);
508 21 : size_t size = poRange.getSize();
509 21 : if (size > INT32_MAX)
510 : {
511 0 : CPLError(CE_Failure, CPLE_OpenFailed, "Invalid size");
512 0 : return FALSE;
513 : }
514 21 : const int nNewLayers = static_cast<int>(size);
515 21 : if (EQUAL(pszFilename, "/vsistdin/"))
516 0 : osBaseLayerName = "layer";
517 42 : CPLString osLayerName;
518 42 : papoLayers = (OGRSelafinLayer **)CPLRealloc(
519 21 : papoLayers, sizeof(void *) * (nLayers + nNewLayers));
520 63 : for (size_t j = 0; j < 2; ++j)
521 : {
522 42 : SelafinTypeDef eType = (j == 0) ? POINTS : ELEMENTS;
523 50 : for (int i = 0; i < poHeader->nSteps; ++i)
524 : {
525 8 : if (poRange.contains(eType, i))
526 : {
527 8 : char szTemp[30] = {};
528 8 : double dfTime = 0.0;
529 8 : if (VSIFSeekL(fp, poHeader->getPosition(i) + 4, SEEK_SET) !=
530 16 : 0 ||
531 8 : Selafin::read_float(fp, dfTime) == 0)
532 : {
533 0 : CPLError(CE_Failure, CPLE_OpenFailed,
534 : "Failed to open %s, wrong format.\n", pszFilename);
535 0 : return FALSE;
536 : }
537 8 : if (poHeader->panStartDate == nullptr)
538 8 : snprintf(szTemp, 29, "%d", i);
539 : else
540 : {
541 : struct tm sDate;
542 0 : memset(&sDate, 0, sizeof(sDate));
543 0 : sDate.tm_year =
544 0 : std::max(poHeader->panStartDate[0], 0) - 1900;
545 0 : sDate.tm_mon = std::max(poHeader->panStartDate[1], 1) - 1;
546 0 : sDate.tm_mday = poHeader->panStartDate[2];
547 0 : sDate.tm_hour = poHeader->panStartDate[3];
548 0 : sDate.tm_min = poHeader->panStartDate[4];
549 0 : double dfSec = poHeader->panStartDate[5] + dfTime;
550 0 : if (dfSec >= 0 && dfSec < 60)
551 0 : sDate.tm_sec = static_cast<int>(dfSec);
552 0 : mktime(&sDate);
553 0 : strftime(szTemp, 29, "%Y_%m_%d_%H_%M_%S", &sDate);
554 : }
555 8 : if (eType == POINTS)
556 4 : osLayerName = osBaseLayerName + "_p" + szTemp;
557 : else
558 4 : osLayerName = osBaseLayerName + "_e" + szTemp;
559 8 : papoLayers[nLayers++] =
560 8 : new OGRSelafinLayer(this, osLayerName, bUpdate,
561 8 : poSpatialRef, poHeader, i, eType);
562 : // poHeader->nRefCount++;
563 : }
564 : }
565 : }
566 :
567 : // Free allocated variables and exit
568 21 : return TRUE;
569 : }
570 :
571 : /************************************************************************/
572 : /* ICreateLayer() */
573 : /************************************************************************/
574 :
575 : OGRLayer *
576 19 : OGRSelafinDataSource::ICreateLayer(const char *pszLayerName,
577 : const OGRGeomFieldDefn *poGeomFieldDefn,
578 : CSLConstList papszOptions)
579 :
580 : {
581 19 : auto eGType = poGeomFieldDefn ? poGeomFieldDefn->GetType() : wkbNone;
582 : const auto poSpatialRefP =
583 19 : poGeomFieldDefn ? poGeomFieldDefn->GetSpatialRef() : nullptr;
584 :
585 19 : CPLDebug("Selafin", "CreateLayer(%s,%s)", pszLayerName,
586 : (eGType == wkbPoint) ? "wkbPoint" : "wkbPolygon");
587 : // Verify we are in update mode.
588 19 : if (!bUpdate)
589 : {
590 0 : CPLError(CE_Failure, CPLE_NoWriteAccess,
591 : "Data source %s opened read-only. "
592 : "New layer %s cannot be created.",
593 : pszName, pszLayerName);
594 0 : return nullptr;
595 : }
596 :
597 : // Check that new layer is a point or polygon layer
598 19 : if (eGType != wkbPoint)
599 : {
600 15 : CPLError(
601 : CE_Failure, CPLE_NoWriteAccess,
602 : "Selafin format can only handle %s layers whereas input is %s\n.",
603 : OGRGeometryTypeToName(wkbPoint), OGRGeometryTypeToName(eGType));
604 15 : return nullptr;
605 : }
606 : // Parse options
607 4 : const char *pszTemp = CSLFetchNameValue(papszOptions, "DATE");
608 4 : const double dfDate = pszTemp != nullptr ? CPLAtof(pszTemp) : 0.0;
609 : // Set the SRS of the datasource if this is the first layer
610 4 : if (nLayers == 0 && poSpatialRefP != nullptr)
611 : {
612 2 : poSpatialRef = poSpatialRefP->Clone();
613 2 : const char *szEpsg = poSpatialRef->GetAttrValue("GEOGCS|AUTHORITY", 1);
614 2 : int nEpsg = 0;
615 2 : if (szEpsg != nullptr)
616 2 : nEpsg = (int)strtol(szEpsg, nullptr, 10);
617 2 : if (nEpsg == 0)
618 : {
619 0 : CPLError(CE_Warning, CPLE_AppDefined,
620 : "Could not find EPSG code for SRS. The SRS won't be saved "
621 : "in the datasource.");
622 : }
623 : else
624 : {
625 2 : poHeader->nEpsg = nEpsg;
626 : }
627 : }
628 : // Create the new layer in the Selafin file by adding a "time step" at the
629 : // end Beware, as the new layer shares the same header, it automatically
630 : // contains the same number of features and fields as the existing ones.
631 : // This may not be intuitive for the user.
632 4 : if (VSIFSeekL(poHeader->fp, 0, SEEK_END) != 0)
633 0 : return nullptr;
634 4 : if (Selafin::write_integer(poHeader->fp, 4) == 0 ||
635 8 : Selafin::write_float(poHeader->fp, dfDate) == 0 ||
636 4 : Selafin::write_integer(poHeader->fp, 4) == 0)
637 : {
638 0 : CPLError(CE_Failure, CPLE_FileIO,
639 : "Could not write to Selafin file %s.\n", pszName);
640 0 : return nullptr;
641 : }
642 4 : double *pdfValues = nullptr;
643 4 : if (poHeader->nPoints > 0)
644 : {
645 : pdfValues =
646 1 : (double *)VSI_MALLOC2_VERBOSE(sizeof(double), poHeader->nPoints);
647 1 : if (pdfValues == nullptr)
648 0 : return nullptr;
649 : }
650 5 : for (int i = 0; i < poHeader->nVar; ++i)
651 : {
652 2 : if (Selafin::write_floatarray(poHeader->fp, pdfValues,
653 1 : poHeader->nPoints) == 0)
654 : {
655 0 : CPLError(CE_Failure, CPLE_FileIO,
656 : "Could not write to Selafin file %s.\n", pszName);
657 0 : CPLFree(pdfValues);
658 0 : return nullptr;
659 : }
660 : }
661 4 : CPLFree(pdfValues);
662 4 : VSIFFlushL(poHeader->fp);
663 4 : poHeader->nSteps++;
664 : // Create two layers as usual, one for points and one for elements
665 4 : nLayers += 2;
666 4 : papoLayers =
667 4 : (OGRSelafinLayer **)CPLRealloc(papoLayers, sizeof(void *) * nLayers);
668 8 : CPLString szName = pszLayerName;
669 4 : CPLString szNewLayerName = szName + "_p";
670 4 : papoLayers[nLayers - 2] =
671 4 : new OGRSelafinLayer(this, szNewLayerName, bUpdate, poSpatialRef,
672 4 : poHeader, poHeader->nSteps - 1, POINTS);
673 4 : szNewLayerName = szName + "_e";
674 4 : papoLayers[nLayers - 1] =
675 4 : new OGRSelafinLayer(this, szNewLayerName, bUpdate, poSpatialRef,
676 4 : poHeader, poHeader->nSteps - 1, ELEMENTS);
677 4 : return papoLayers[nLayers - 2];
678 : }
679 :
680 : /************************************************************************/
681 : /* DeleteLayer() */
682 : /************************************************************************/
683 1 : OGRErr OGRSelafinDataSource::DeleteLayer(int iLayer)
684 : {
685 : // Verify we are in update mode.
686 1 : if (!bUpdate)
687 : {
688 0 : CPLError(CE_Failure, CPLE_NoWriteAccess,
689 : "Data source %s opened read-only. "
690 : "Layer %d cannot be deleted.\n",
691 : pszName, iLayer);
692 0 : return OGRERR_FAILURE;
693 : }
694 1 : if (iLayer < 0 || iLayer >= nLayers)
695 : {
696 0 : CPLError(CE_Failure, CPLE_AppDefined,
697 : "Layer %d not in legal range of 0 to %d.", iLayer,
698 0 : nLayers - 1);
699 0 : return OGRERR_FAILURE;
700 : }
701 : // Delete layer in file. Here we don't need to create a copy of the file
702 : // because we only update values and it can't get corrupted even if the
703 : // system crashes during the operation
704 1 : const int nNum = papoLayers[iLayer]->GetStepNumber();
705 1 : double *dfValues = nullptr;
706 2 : for (int i = nNum; i < poHeader->nSteps - 1; ++i)
707 : {
708 1 : double dfTime = 0.0;
709 1 : if (VSIFSeekL(poHeader->fp, poHeader->getPosition(i + 1) + 4,
710 1 : SEEK_SET) != 0 ||
711 1 : Selafin::read_float(poHeader->fp, dfTime) == 0 ||
712 1 : VSIFSeekL(poHeader->fp, poHeader->getPosition(i) + 4, SEEK_SET) !=
713 2 : 0 ||
714 1 : Selafin::write_float(poHeader->fp, dfTime) == 0)
715 : {
716 0 : CPLError(CE_Failure, CPLE_FileIO,
717 : "Could not update Selafin file %s.\n", pszName);
718 0 : return OGRERR_FAILURE;
719 : }
720 2 : for (int j = 0; j < poHeader->nVar; ++j)
721 : {
722 1 : bool ok = true;
723 1 : if (VSIFSeekL(poHeader->fp, poHeader->getPosition(i + 1) + 12,
724 1 : SEEK_SET) != 0)
725 : {
726 0 : ok = false;
727 : }
728 : else
729 : {
730 2 : int ret = Selafin::read_floatarray(poHeader->fp, &dfValues,
731 1 : poHeader->nFileSize);
732 1 : if (ret < 0 || ret != poHeader->nPoints ||
733 1 : VSIFSeekL(poHeader->fp, poHeader->getPosition(i) + 12,
734 2 : SEEK_SET) != 0 ||
735 1 : Selafin::write_floatarray(poHeader->fp, dfValues,
736 1 : poHeader->nPoints) == 0)
737 : {
738 0 : ok = false;
739 : }
740 : }
741 1 : if (!ok)
742 : {
743 0 : CPLError(CE_Failure, CPLE_FileIO,
744 : "Could not update Selafin file %s.\n", pszName);
745 0 : CPLFree(dfValues);
746 0 : return OGRERR_FAILURE;
747 : }
748 1 : CPLFree(dfValues);
749 1 : dfValues = nullptr;
750 : }
751 : }
752 : // Delete all layers with the same step number in layer list. Usually there
753 : // are two of them: one for points and one for elements, but we can't rely
754 : // on that because of possible layer filtering specifications
755 5 : for (int i = 0; i < nLayers; ++i)
756 : {
757 4 : if (papoLayers[i]->GetStepNumber() == nNum)
758 : {
759 2 : delete papoLayers[i];
760 2 : nLayers--;
761 7 : for (int j = i; j < nLayers; ++j)
762 5 : papoLayers[j] = papoLayers[j + 1];
763 2 : --i;
764 : }
765 : }
766 1 : return OGRERR_NONE;
767 : }
|