Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: Natural Resources Canada's Geoid BYN file format
4 : * Purpose: Implementation of BYN format
5 : * Author: Ivan Lucena, ivan.lucena@outlook.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2018, Ivan Lucena
9 : * Copyright (c) 2018, Even Rouault
10 : *
11 : * SPDX-License-Identifier: MIT
12 : ****************************************************************************/
13 :
14 : #include "cpl_port.h"
15 : #include "byndataset.h"
16 : #include "rawdataset.h"
17 :
18 : #include "cpl_string.h"
19 : #include "gdal_frmts.h"
20 : #include "ogr_spatialref.h"
21 : #include "ogr_srs_api.h"
22 :
23 : #include <algorithm>
24 : #include <cstdlib>
25 : #include <limits>
26 :
27 : // Specification at
28 : // https://www.nrcan.gc.ca/sites/www.nrcan.gc.ca/files/earthsciences/pdf/gpshgrid_e.pdf
29 :
30 : const static BYNEllipsoids EllipsoidTable[] = {
31 : {"GRS80", 6378137.0, 298.257222101},
32 : {"WGS84", 6378137.0, 298.257223564},
33 : {"ALT1", 6378136.3, 298.256415099},
34 : {"GRS67", 6378160.0, 298.247167427},
35 : {"ELLIP1", 6378136.46, 298.256415099},
36 : {"ALT2", 6378136.3, 298.257},
37 : {"ELLIP2", 6378136.0, 298.257},
38 : {"CLARKE 1866", 6378206.4, 294.9786982}};
39 :
40 : /************************************************************************/
41 : /* BYNRasterBand() */
42 : /************************************************************************/
43 :
44 8 : BYNRasterBand::BYNRasterBand(GDALDataset *poDSIn, int nBandIn,
45 : VSILFILE *fpRawIn, vsi_l_offset nImgOffsetIn,
46 : int nPixelOffsetIn, int nLineOffsetIn,
47 8 : GDALDataType eDataTypeIn, int bNativeOrderIn)
48 : : RawRasterBand(poDSIn, nBandIn, fpRawIn, nImgOffsetIn, nPixelOffsetIn,
49 : nLineOffsetIn, eDataTypeIn, bNativeOrderIn,
50 8 : RawRasterBand::OwnFP::NO)
51 : {
52 8 : }
53 :
54 : /************************************************************************/
55 : /* ~BYNRasterBand() */
56 : /************************************************************************/
57 :
58 16 : BYNRasterBand::~BYNRasterBand()
59 : {
60 16 : }
61 :
62 : /************************************************************************/
63 : /* GetNoDataValue() */
64 : /************************************************************************/
65 :
66 6 : double BYNRasterBand::GetNoDataValue(int *pbSuccess)
67 : {
68 6 : if (pbSuccess)
69 5 : *pbSuccess = TRUE;
70 6 : int bSuccess = FALSE;
71 6 : double dfNoData = GDALPamRasterBand::GetNoDataValue(&bSuccess);
72 6 : if (bSuccess)
73 : {
74 2 : return dfNoData;
75 : }
76 4 : const double dfFactor =
77 4 : reinterpret_cast<BYNDataset *>(poDS)->hHeader.dfFactor;
78 4 : return eDataType == GDT_Int16 ? 32767.0 : 9999.0 * dfFactor;
79 : }
80 :
81 : /************************************************************************/
82 : /* GetScale() */
83 : /************************************************************************/
84 :
85 1 : double BYNRasterBand::GetScale(int *pbSuccess)
86 : {
87 1 : if (pbSuccess != nullptr)
88 1 : *pbSuccess = TRUE;
89 1 : const double dfFactor =
90 1 : reinterpret_cast<BYNDataset *>(poDS)->hHeader.dfFactor;
91 1 : return (dfFactor != 0.0) ? 1.0 / dfFactor : 0.0;
92 : }
93 :
94 : /************************************************************************/
95 : /* SetScale() */
96 : /************************************************************************/
97 :
98 1 : CPLErr BYNRasterBand::SetScale(double dfNewValue)
99 : {
100 1 : BYNDataset *poIDS = reinterpret_cast<BYNDataset *>(poDS);
101 1 : poIDS->hHeader.dfFactor = 1.0 / dfNewValue;
102 1 : return CE_None;
103 : }
104 :
105 : /************************************************************************/
106 : /* ==================================================================== */
107 : /* BYNDataset */
108 : /* ==================================================================== */
109 : /************************************************************************/
110 :
111 8 : BYNDataset::BYNDataset()
112 : : fpImage(nullptr), hHeader{0, 0, 0, 0, 0, 0, 0, 0, 0.0, 0, 0, 0,
113 8 : 0, 0, 0, 0, 0, 0.0, 0.0, 0, 0, 0.0, 0}
114 : {
115 8 : m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
116 8 : adfGeoTransform[0] = 0.0;
117 8 : adfGeoTransform[1] = 1.0;
118 8 : adfGeoTransform[2] = 0.0;
119 8 : adfGeoTransform[3] = 0.0;
120 8 : adfGeoTransform[4] = 0.0;
121 8 : adfGeoTransform[5] = 1.0;
122 8 : }
123 :
124 : /************************************************************************/
125 : /* ~BYNDataset() */
126 : /************************************************************************/
127 :
128 16 : BYNDataset::~BYNDataset()
129 :
130 : {
131 8 : BYNDataset::Close();
132 16 : }
133 :
134 : /************************************************************************/
135 : /* Close() */
136 : /************************************************************************/
137 :
138 16 : CPLErr BYNDataset::Close()
139 : {
140 16 : CPLErr eErr = CE_None;
141 16 : if (nOpenFlags != OPEN_FLAGS_CLOSED)
142 : {
143 8 : if (BYNDataset::FlushCache(true) != CE_None)
144 0 : eErr = CE_Failure;
145 :
146 8 : if (GetAccess() == GA_Update)
147 1 : UpdateHeader();
148 :
149 8 : if (fpImage != nullptr)
150 : {
151 8 : if (VSIFCloseL(fpImage) != 0)
152 : {
153 0 : eErr = CE_Failure;
154 0 : CPLError(CE_Failure, CPLE_FileIO, "I/O error");
155 : }
156 : }
157 :
158 8 : if (GDALPamDataset::Close() != CE_None)
159 0 : eErr = CE_Failure;
160 : }
161 16 : return eErr;
162 : }
163 :
164 : /************************************************************************/
165 : /* Identify() */
166 : /************************************************************************/
167 :
168 51863 : int BYNDataset::Identify(GDALOpenInfo *poOpenInfo)
169 :
170 : {
171 51863 : if (poOpenInfo->nHeaderBytes < BYN_HDR_SZ)
172 48562 : return FALSE;
173 :
174 : /* -------------------------------------------------------------------- */
175 : /* Check file extension (.byn/.err) */
176 : /* -------------------------------------------------------------------- */
177 : #ifndef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
178 3301 : const char *pszFileExtension = poOpenInfo->osExtension.c_str();
179 :
180 3301 : if (!EQUAL(pszFileExtension, "byn") && !EQUAL(pszFileExtension, "err"))
181 : {
182 3285 : return FALSE;
183 : }
184 : #endif
185 :
186 : /* -------------------------------------------------------------------- */
187 : /* Check some value's ranges on header */
188 : /* -------------------------------------------------------------------- */
189 :
190 16 : BYNHeader hHeader = {0, 0, 0, 0, 0, 0, 0, 0, 0.0, 0, 0, 0,
191 : 0, 0, 0, 0, 0, 0.0, 0.0, 0, 0, 0.0, 0};
192 :
193 16 : buffer2header(poOpenInfo->pabyHeader, &hHeader);
194 :
195 16 : if (hHeader.nGlobal < 0 || hHeader.nGlobal > 1 || hHeader.nType < 0 ||
196 16 : hHeader.nType > 9 || (hHeader.nSizeOf != 2 && hHeader.nSizeOf != 4) ||
197 16 : hHeader.nVDatum < 0 || hHeader.nVDatum > 3 || hHeader.nDescrip < 0 ||
198 16 : hHeader.nDescrip > 3 || hHeader.nSubType < 0 || hHeader.nSubType > 9 ||
199 16 : hHeader.nDatum < 0 || hHeader.nDatum > 1 || hHeader.nEllipsoid < 0 ||
200 16 : hHeader.nEllipsoid > 7 || hHeader.nByteOrder < 0 ||
201 16 : hHeader.nByteOrder > 1 || hHeader.nScale < 0 || hHeader.nScale > 1)
202 0 : return FALSE;
203 :
204 : #if 0
205 : // We have disabled those checks as invalid values are often found in some
206 : // datasets, such as http://s3.microsurvey.com/os/fieldgenius/geoids/Lithuania.zip
207 : // We don't use those fields, so we may just ignore them.
208 : if((hHeader.nTideSys < 0 || hHeader.nTideSys > 2 ||
209 : hHeader.nPtType < 0 || hHeader.nPtType > 1 ))
210 : {
211 : // Some datasets use 0xCC as a marker for invalidity for
212 : // records starting from Geopotential Wo
213 : for( int i = 52; i < 78; i++ )
214 : {
215 : if( poOpenInfo->pabyHeader[i] != 0xCC )
216 : return FALSE;
217 : }
218 : }
219 : #endif
220 :
221 16 : if (hHeader.nScale == 0)
222 : {
223 32 : if ((std::abs(static_cast<GIntBig>(hHeader.nSouth) -
224 32 : (hHeader.nDLat / 2)) > BYN_MAX_LAT) ||
225 16 : (std::abs(static_cast<GIntBig>(hHeader.nNorth) +
226 32 : (hHeader.nDLat / 2)) > BYN_MAX_LAT) ||
227 16 : (std::abs(static_cast<GIntBig>(hHeader.nWest) -
228 48 : (hHeader.nDLon / 2)) > BYN_MAX_LON) ||
229 16 : (std::abs(static_cast<GIntBig>(hHeader.nEast) +
230 16 : (hHeader.nDLon / 2)) > BYN_MAX_LON))
231 0 : return FALSE;
232 : }
233 : else
234 : {
235 0 : if ((std::abs(static_cast<GIntBig>(hHeader.nSouth) -
236 0 : (hHeader.nDLat / 2)) > BYN_MAX_LAT_SCL) ||
237 0 : (std::abs(static_cast<GIntBig>(hHeader.nNorth) +
238 0 : (hHeader.nDLat / 2)) > BYN_MAX_LAT_SCL) ||
239 0 : (std::abs(static_cast<GIntBig>(hHeader.nWest) -
240 0 : (hHeader.nDLon / 2)) > BYN_MAX_LON_SCL) ||
241 0 : (std::abs(static_cast<GIntBig>(hHeader.nEast) +
242 0 : (hHeader.nDLon / 2)) > BYN_MAX_LON_SCL))
243 0 : return FALSE;
244 : }
245 :
246 16 : return TRUE;
247 : }
248 :
249 : /************************************************************************/
250 : /* Open() */
251 : /************************************************************************/
252 :
253 8 : GDALDataset *BYNDataset::Open(GDALOpenInfo *poOpenInfo)
254 :
255 : {
256 8 : if (!Identify(poOpenInfo) || poOpenInfo->fpL == nullptr)
257 0 : return nullptr;
258 :
259 : /* -------------------------------------------------------------------- */
260 : /* Create a corresponding GDALDataset. */
261 : /* -------------------------------------------------------------------- */
262 :
263 16 : auto poDS = std::make_unique<BYNDataset>();
264 :
265 8 : poDS->eAccess = poOpenInfo->eAccess;
266 8 : std::swap(poDS->fpImage, poOpenInfo->fpL);
267 :
268 : /* -------------------------------------------------------------------- */
269 : /* Read the header. */
270 : /* -------------------------------------------------------------------- */
271 :
272 8 : buffer2header(poOpenInfo->pabyHeader, &poDS->hHeader);
273 :
274 : /********************************/
275 : /* Scale boundaries and spacing */
276 : /********************************/
277 :
278 8 : double dfSouth = poDS->hHeader.nSouth;
279 8 : double dfNorth = poDS->hHeader.nNorth;
280 8 : double dfWest = poDS->hHeader.nWest;
281 8 : double dfEast = poDS->hHeader.nEast;
282 8 : double dfDLat = poDS->hHeader.nDLat;
283 8 : double dfDLon = poDS->hHeader.nDLon;
284 :
285 8 : if (poDS->hHeader.nScale == 1)
286 : {
287 0 : dfSouth *= BYN_SCALE;
288 0 : dfNorth *= BYN_SCALE;
289 0 : dfWest *= BYN_SCALE;
290 0 : dfEast *= BYN_SCALE;
291 0 : dfDLat *= BYN_SCALE;
292 0 : dfDLon *= BYN_SCALE;
293 : }
294 :
295 : /******************************/
296 : /* Calculate rows and columns */
297 : /******************************/
298 :
299 8 : double dfXSize = -1;
300 8 : double dfYSize = -1;
301 :
302 8 : poDS->nRasterXSize = -1;
303 8 : poDS->nRasterYSize = -1;
304 :
305 8 : if (dfDLat != 0.0 && dfDLon != 0.0)
306 : {
307 8 : dfXSize = ((dfEast - dfWest + 1.0) / dfDLon) + 1.0;
308 8 : dfYSize = ((dfNorth - dfSouth + 1.0) / dfDLat) + 1.0;
309 : }
310 :
311 8 : if (dfXSize > 0.0 && dfXSize < std::numeric_limits<double>::max() &&
312 16 : dfYSize > 0.0 && dfYSize < std::numeric_limits<double>::max())
313 : {
314 8 : poDS->nRasterXSize = static_cast<GInt32>(dfXSize);
315 8 : poDS->nRasterYSize = static_cast<GInt32>(dfYSize);
316 : }
317 :
318 8 : if (!GDALCheckDatasetDimensions(poDS->nRasterXSize, poDS->nRasterYSize))
319 : {
320 0 : return nullptr;
321 : }
322 :
323 : /*****************************/
324 : /* Build GeoTransform matrix */
325 : /*****************************/
326 :
327 8 : poDS->adfGeoTransform[0] = (dfWest - (dfDLon / 2.0)) / 3600.0;
328 8 : poDS->adfGeoTransform[1] = dfDLon / 3600.0;
329 8 : poDS->adfGeoTransform[2] = 0.0;
330 8 : poDS->adfGeoTransform[3] = (dfNorth + (dfDLat / 2.0)) / 3600.0;
331 8 : poDS->adfGeoTransform[4] = 0.0;
332 8 : poDS->adfGeoTransform[5] = -1 * dfDLat / 3600.0;
333 :
334 : /*********************/
335 : /* Set data type */
336 : /*********************/
337 :
338 8 : GDALDataType eDT = GDT_Unknown;
339 :
340 8 : if (poDS->hHeader.nSizeOf == 2)
341 0 : eDT = GDT_Int16;
342 8 : else if (poDS->hHeader.nSizeOf == 4)
343 8 : eDT = GDT_Int32;
344 : else
345 : {
346 0 : return nullptr;
347 : }
348 :
349 : /* -------------------------------------------------------------------- */
350 : /* Create band information object. */
351 : /* -------------------------------------------------------------------- */
352 :
353 8 : const int nDTSize = GDALGetDataTypeSizeBytes(eDT);
354 :
355 8 : int bIsLSB = poDS->hHeader.nByteOrder == 1 ? 1 : 0;
356 :
357 : auto poBand = std::make_unique<BYNRasterBand>(
358 8 : poDS.get(), 1, poDS->fpImage, BYN_HDR_SZ, nDTSize,
359 24 : poDS->nRasterXSize * nDTSize, eDT, CPL_IS_LSB == bIsLSB);
360 8 : if (!poBand->IsValid())
361 0 : return nullptr;
362 8 : poDS->SetBand(1, std::move(poBand));
363 :
364 : /* -------------------------------------------------------------------- */
365 : /* Initialize any PAM information. */
366 : /* -------------------------------------------------------------------- */
367 :
368 8 : poDS->SetDescription(poOpenInfo->pszFilename);
369 8 : poDS->TryLoadXML();
370 :
371 : /* -------------------------------------------------------------------- */
372 : /* Check for overviews. */
373 : /* -------------------------------------------------------------------- */
374 :
375 8 : poDS->oOvManager.Initialize(poDS.get(), poOpenInfo->pszFilename);
376 :
377 8 : return poDS.release();
378 : }
379 :
380 : /************************************************************************/
381 : /* GetGeoTransform() */
382 : /************************************************************************/
383 :
384 2 : CPLErr BYNDataset::GetGeoTransform(double *padfTransform)
385 :
386 : {
387 2 : memcpy(padfTransform, adfGeoTransform, sizeof(double) * 6);
388 2 : return CE_None;
389 : }
390 :
391 : /************************************************************************/
392 : /* SetGeoTransform() */
393 : /************************************************************************/
394 :
395 1 : CPLErr BYNDataset::SetGeoTransform(double *padfTransform)
396 :
397 : {
398 1 : if (padfTransform[2] != 0.0 || padfTransform[4] != 0.0)
399 : {
400 0 : CPLError(CE_Failure, CPLE_AppDefined,
401 : "Attempt to write skewed or rotated geotransform to byn.");
402 0 : return CE_Failure;
403 : }
404 1 : memcpy(adfGeoTransform, padfTransform, sizeof(double) * 6);
405 1 : return CE_None;
406 : }
407 :
408 : /************************************************************************/
409 : /* GetSpatialRef() */
410 : /************************************************************************/
411 :
412 1 : const OGRSpatialReference *BYNDataset::GetSpatialRef() const
413 :
414 : {
415 1 : if (!m_oSRS.IsEmpty())
416 0 : return &m_oSRS;
417 :
418 : /* Try to use a prefefined EPSG compound CS */
419 :
420 1 : if (hHeader.nDatum == 1 && hHeader.nVDatum == 2)
421 : {
422 0 : m_oSRS.importFromEPSG(BYN_DATUM_1_VDATUM_2);
423 0 : return &m_oSRS;
424 : }
425 :
426 : /* Build the GEOGCS based on Datum ( or Ellipsoid )*/
427 :
428 1 : bool bNoGeogCS = false;
429 :
430 1 : if (hHeader.nDatum == 0)
431 1 : m_oSRS.importFromEPSG(BYN_DATUM_0);
432 0 : else if (hHeader.nDatum == 1)
433 0 : m_oSRS.importFromEPSG(BYN_DATUM_1);
434 : else
435 : {
436 : /* Build GEOGCS based on Ellipsoid (Table 3) */
437 :
438 0 : if (hHeader.nEllipsoid > -1 &&
439 0 : hHeader.nEllipsoid <
440 : static_cast<GInt16>(CPL_ARRAYSIZE(EllipsoidTable)))
441 0 : m_oSRS.SetGeogCS(
442 0 : CPLSPrintf("BYN Ellipsoid(%d)", hHeader.nEllipsoid),
443 0 : "Unspecified", EllipsoidTable[hHeader.nEllipsoid].pszName,
444 0 : EllipsoidTable[hHeader.nEllipsoid].dfSemiMajor,
445 0 : EllipsoidTable[hHeader.nEllipsoid].dfInvFlattening);
446 : else
447 0 : bNoGeogCS = true;
448 : }
449 :
450 : /* Build the VERT_CS based on VDatum */
451 :
452 2 : OGRSpatialReference oSRSComp;
453 2 : OGRSpatialReference oSRSVert;
454 :
455 1 : int nVertCS = 0;
456 :
457 1 : if (hHeader.nVDatum == 1)
458 0 : nVertCS = BYN_VDATUM_1;
459 1 : else if (hHeader.nVDatum == 2)
460 1 : nVertCS = BYN_VDATUM_2;
461 0 : else if (hHeader.nVDatum == 3)
462 0 : nVertCS = BYN_VDATUM_3;
463 : else
464 : {
465 : /* Return GEOGCS ( .err files ) */
466 :
467 0 : if (bNoGeogCS)
468 0 : return nullptr;
469 :
470 0 : return &m_oSRS;
471 : }
472 :
473 1 : oSRSVert.importFromEPSG(nVertCS);
474 :
475 : /* Create CPMPD_CS with GEOGCS and VERT_CS */
476 :
477 1 : if (oSRSComp.SetCompoundCS(CPLSPrintf("BYN Datum(%d) & VDatum(%d)",
478 1 : hHeader.nDatum, hHeader.nDatum),
479 2 : &m_oSRS, &oSRSVert) == CE_None)
480 : {
481 : /* Return COMPD_CS with GEOGCS and VERT_CS */
482 :
483 1 : m_oSRS = std::move(oSRSComp);
484 1 : m_oSRS.SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
485 1 : return &m_oSRS;
486 : }
487 :
488 0 : return nullptr;
489 : }
490 :
491 : /************************************************************************/
492 : /* SetSpatialRef() */
493 : /************************************************************************/
494 :
495 1 : CPLErr BYNDataset::SetSpatialRef(const OGRSpatialReference *poSRS)
496 :
497 : {
498 1 : if (poSRS == nullptr)
499 0 : return CE_Failure;
500 :
501 1 : m_oSRS = *poSRS;
502 :
503 : /* Try to recognize prefefined EPSG compound CS */
504 :
505 1 : if (poSRS->IsCompound())
506 : {
507 1 : const char *pszAuthName = poSRS->GetAuthorityName("COMPD_CS");
508 1 : const char *pszAuthCode = poSRS->GetAuthorityCode("COMPD_CS");
509 :
510 1 : if (pszAuthName != nullptr && pszAuthCode != nullptr &&
511 0 : EQUAL(pszAuthName, "EPSG") &&
512 0 : atoi(pszAuthCode) == BYN_DATUM_1_VDATUM_2)
513 : {
514 0 : hHeader.nVDatum = 2;
515 0 : hHeader.nDatum = 1;
516 0 : return CE_None;
517 : }
518 : }
519 :
520 1 : OGRSpatialReference oSRSTemp;
521 :
522 : /* Try to match GEOGCS */
523 :
524 1 : if (poSRS->IsGeographic())
525 : {
526 1 : oSRSTemp.importFromEPSG(BYN_DATUM_0);
527 1 : if (poSRS->IsSameGeogCS(&oSRSTemp))
528 1 : hHeader.nDatum = 0;
529 : else
530 : {
531 0 : oSRSTemp.importFromEPSG(BYN_DATUM_1);
532 0 : if (poSRS->IsSameGeogCS(&oSRSTemp))
533 0 : hHeader.nDatum = 1;
534 : }
535 : }
536 :
537 : /* Try to match VERT_CS */
538 :
539 1 : if (poSRS->IsVertical())
540 : {
541 1 : oSRSTemp.importFromEPSG(BYN_VDATUM_1);
542 1 : if (poSRS->IsSameVertCS(&oSRSTemp))
543 0 : hHeader.nVDatum = 1;
544 : else
545 : {
546 1 : oSRSTemp.importFromEPSG(BYN_VDATUM_2);
547 1 : if (poSRS->IsSameVertCS(&oSRSTemp))
548 1 : hHeader.nVDatum = 2;
549 : else
550 : {
551 0 : oSRSTemp.importFromEPSG(BYN_VDATUM_3);
552 0 : if (poSRS->IsSameVertCS(&oSRSTemp))
553 0 : hHeader.nVDatum = 3;
554 : }
555 : }
556 : }
557 :
558 1 : return CE_None;
559 : }
560 :
561 : /*----------------------------------------------------------------------*/
562 : /* buffer2header() */
563 : /*----------------------------------------------------------------------*/
564 :
565 24 : void BYNDataset::buffer2header(const GByte *pabyBuf, BYNHeader *pohHeader)
566 :
567 : {
568 24 : memcpy(&pohHeader->nSouth, pabyBuf, 4);
569 24 : memcpy(&pohHeader->nNorth, pabyBuf + 4, 4);
570 24 : memcpy(&pohHeader->nWest, pabyBuf + 8, 4);
571 24 : memcpy(&pohHeader->nEast, pabyBuf + 12, 4);
572 24 : memcpy(&pohHeader->nDLat, pabyBuf + 16, 2);
573 24 : memcpy(&pohHeader->nDLon, pabyBuf + 18, 2);
574 24 : memcpy(&pohHeader->nGlobal, pabyBuf + 20, 2);
575 24 : memcpy(&pohHeader->nType, pabyBuf + 22, 2);
576 24 : memcpy(&pohHeader->dfFactor, pabyBuf + 24, 8);
577 24 : memcpy(&pohHeader->nSizeOf, pabyBuf + 32, 2);
578 24 : memcpy(&pohHeader->nVDatum, pabyBuf + 34, 2);
579 24 : memcpy(&pohHeader->nDescrip, pabyBuf + 40, 2);
580 24 : memcpy(&pohHeader->nSubType, pabyBuf + 42, 2);
581 24 : memcpy(&pohHeader->nDatum, pabyBuf + 44, 2);
582 24 : memcpy(&pohHeader->nEllipsoid, pabyBuf + 46, 2);
583 24 : memcpy(&pohHeader->nByteOrder, pabyBuf + 48, 2);
584 24 : memcpy(&pohHeader->nScale, pabyBuf + 50, 2);
585 24 : memcpy(&pohHeader->dfWo, pabyBuf + 52, 8);
586 24 : memcpy(&pohHeader->dfGM, pabyBuf + 60, 8);
587 24 : memcpy(&pohHeader->nTideSys, pabyBuf + 68, 2);
588 24 : memcpy(&pohHeader->nRealiz, pabyBuf + 70, 2);
589 24 : memcpy(&pohHeader->dEpoch, pabyBuf + 72, 4);
590 24 : memcpy(&pohHeader->nPtType, pabyBuf + 76, 2);
591 :
592 : #if defined(CPL_MSB)
593 : CPL_LSBPTR32(&pohHeader->nSouth);
594 : CPL_LSBPTR32(&pohHeader->nNorth);
595 : CPL_LSBPTR32(&pohHeader->nWest);
596 : CPL_LSBPTR32(&pohHeader->nEast);
597 : CPL_LSBPTR16(&pohHeader->nDLat);
598 : CPL_LSBPTR16(&pohHeader->nDLon);
599 : CPL_LSBPTR16(&pohHeader->nGlobal);
600 : CPL_LSBPTR16(&pohHeader->nType);
601 : CPL_LSBPTR64(&pohHeader->dfFactor);
602 : CPL_LSBPTR16(&pohHeader->nSizeOf);
603 : CPL_LSBPTR16(&pohHeader->nVDatum);
604 : CPL_LSBPTR16(&pohHeader->nDescrip);
605 : CPL_LSBPTR16(&pohHeader->nSubType);
606 : CPL_LSBPTR16(&pohHeader->nDatum);
607 : CPL_LSBPTR16(&pohHeader->nEllipsoid);
608 : CPL_LSBPTR16(&pohHeader->nByteOrder);
609 : CPL_LSBPTR16(&pohHeader->nScale);
610 : CPL_LSBPTR64(&pohHeader->dfWo);
611 : CPL_LSBPTR64(&pohHeader->dfGM);
612 : CPL_LSBPTR16(&pohHeader->nTideSys);
613 : CPL_LSBPTR16(&pohHeader->nRealiz);
614 : CPL_LSBPTR32(&pohHeader->dEpoch);
615 : CPL_LSBPTR16(&pohHeader->nPtType);
616 : #endif
617 :
618 : #if DEBUG
619 24 : CPLDebug("BYN", "South = %d", pohHeader->nSouth);
620 24 : CPLDebug("BYN", "North = %d", pohHeader->nNorth);
621 24 : CPLDebug("BYN", "West = %d", pohHeader->nWest);
622 24 : CPLDebug("BYN", "East = %d", pohHeader->nEast);
623 24 : CPLDebug("BYN", "DLat = %d", pohHeader->nDLat);
624 24 : CPLDebug("BYN", "DLon = %d", pohHeader->nDLon);
625 24 : CPLDebug("BYN", "DGlobal = %d", pohHeader->nGlobal);
626 24 : CPLDebug("BYN", "DType = %d", pohHeader->nType);
627 24 : CPLDebug("BYN", "Factor = %f", pohHeader->dfFactor);
628 24 : CPLDebug("BYN", "SizeOf = %d", pohHeader->nSizeOf);
629 24 : CPLDebug("BYN", "VDatum = %d", pohHeader->nVDatum);
630 24 : CPLDebug("BYN", "Data = %d", pohHeader->nDescrip);
631 24 : CPLDebug("BYN", "SubType = %d", pohHeader->nSubType);
632 24 : CPLDebug("BYN", "Datum = %d", pohHeader->nDatum);
633 24 : CPLDebug("BYN", "Ellipsoid = %d", pohHeader->nEllipsoid);
634 24 : CPLDebug("BYN", "ByteOrder = %d", pohHeader->nByteOrder);
635 24 : CPLDebug("BYN", "Scale = %d", pohHeader->nScale);
636 24 : CPLDebug("BYN", "Wo = %f", pohHeader->dfWo);
637 24 : CPLDebug("BYN", "GM = %f", pohHeader->dfGM);
638 24 : CPLDebug("BYN", "TideSystem = %d", pohHeader->nTideSys);
639 24 : CPLDebug("BYN", "RefRealzation = %d", pohHeader->nRealiz);
640 24 : CPLDebug("BYN", "Epoch = %f", pohHeader->dEpoch);
641 24 : CPLDebug("BYN", "PtType = %d", pohHeader->nPtType);
642 : #endif
643 24 : }
644 :
645 : /*----------------------------------------------------------------------*/
646 : /* header2buffer() */
647 : /*----------------------------------------------------------------------*/
648 :
649 2 : void BYNDataset::header2buffer(const BYNHeader *pohHeader, GByte *pabyBuf)
650 :
651 : {
652 2 : memcpy(pabyBuf, &pohHeader->nSouth, 4);
653 2 : memcpy(pabyBuf + 4, &pohHeader->nNorth, 4);
654 2 : memcpy(pabyBuf + 8, &pohHeader->nWest, 4);
655 2 : memcpy(pabyBuf + 12, &pohHeader->nEast, 4);
656 2 : memcpy(pabyBuf + 16, &pohHeader->nDLat, 2);
657 2 : memcpy(pabyBuf + 18, &pohHeader->nDLon, 2);
658 2 : memcpy(pabyBuf + 20, &pohHeader->nGlobal, 2);
659 2 : memcpy(pabyBuf + 22, &pohHeader->nType, 2);
660 2 : memcpy(pabyBuf + 24, &pohHeader->dfFactor, 8);
661 2 : memcpy(pabyBuf + 32, &pohHeader->nSizeOf, 2);
662 2 : memcpy(pabyBuf + 34, &pohHeader->nVDatum, 2);
663 2 : memcpy(pabyBuf + 40, &pohHeader->nDescrip, 2);
664 2 : memcpy(pabyBuf + 42, &pohHeader->nSubType, 2);
665 2 : memcpy(pabyBuf + 44, &pohHeader->nDatum, 2);
666 2 : memcpy(pabyBuf + 46, &pohHeader->nEllipsoid, 2);
667 2 : memcpy(pabyBuf + 48, &pohHeader->nByteOrder, 2);
668 2 : memcpy(pabyBuf + 50, &pohHeader->nScale, 2);
669 2 : memcpy(pabyBuf + 52, &pohHeader->dfWo, 8);
670 2 : memcpy(pabyBuf + 60, &pohHeader->dfGM, 8);
671 2 : memcpy(pabyBuf + 68, &pohHeader->nTideSys, 2);
672 2 : memcpy(pabyBuf + 70, &pohHeader->nRealiz, 2);
673 2 : memcpy(pabyBuf + 72, &pohHeader->dEpoch, 4);
674 2 : memcpy(pabyBuf + 76, &pohHeader->nPtType, 2);
675 :
676 : #if defined(CPL_MSB)
677 : CPL_LSBPTR32(pabyBuf);
678 : CPL_LSBPTR32(pabyBuf + 4);
679 : CPL_LSBPTR32(pabyBuf + 8);
680 : CPL_LSBPTR32(pabyBuf + 12);
681 : CPL_LSBPTR16(pabyBuf + 16);
682 : CPL_LSBPTR16(pabyBuf + 18);
683 : CPL_LSBPTR16(pabyBuf + 20);
684 : CPL_LSBPTR16(pabyBuf + 22);
685 : CPL_LSBPTR64(pabyBuf + 24);
686 : CPL_LSBPTR16(pabyBuf + 32);
687 : CPL_LSBPTR16(pabyBuf + 34);
688 : CPL_LSBPTR16(pabyBuf + 40);
689 : CPL_LSBPTR16(pabyBuf + 42);
690 : CPL_LSBPTR16(pabyBuf + 44);
691 : CPL_LSBPTR16(pabyBuf + 46);
692 : CPL_LSBPTR16(pabyBuf + 48);
693 : CPL_LSBPTR16(pabyBuf + 50);
694 : CPL_LSBPTR64(pabyBuf + 52);
695 : CPL_LSBPTR64(pabyBuf + 60);
696 : CPL_LSBPTR16(pabyBuf + 68);
697 : CPL_LSBPTR16(pabyBuf + 70);
698 : CPL_LSBPTR32(pabyBuf + 72);
699 : CPL_LSBPTR16(pabyBuf + 76);
700 : #endif
701 2 : }
702 :
703 : /************************************************************************/
704 : /* Create() */
705 : /************************************************************************/
706 :
707 50 : GDALDataset *BYNDataset::Create(const char *pszFilename, int nXSize, int nYSize,
708 : int /* nBands */, GDALDataType eType,
709 : char ** /* papszOptions */)
710 : {
711 50 : if (eType != GDT_Int16 && eType != GDT_Int32)
712 : {
713 43 : CPLError(CE_Failure, CPLE_AppDefined,
714 : "Attempt to create byn file with unsupported data type '%s'.",
715 : GDALGetDataTypeName(eType));
716 43 : return nullptr;
717 : }
718 :
719 : /* -------------------------------------------------------------------- */
720 : /* Check file extension (.byn/.err) */
721 : /* -------------------------------------------------------------------- */
722 :
723 14 : const std::string osExt = CPLGetExtensionSafe(pszFilename);
724 :
725 7 : if (!EQUAL(osExt.c_str(), "byn") && !EQUAL(osExt.c_str(), "err"))
726 : {
727 6 : CPLError(
728 : CE_Failure, CPLE_AppDefined,
729 : "Attempt to create byn file with extension other than byn/err.");
730 6 : return nullptr;
731 : }
732 :
733 : /* -------------------------------------------------------------------- */
734 : /* Try to create the file. */
735 : /* -------------------------------------------------------------------- */
736 :
737 1 : VSILFILE *fp = VSIFOpenL(pszFilename, "wb+");
738 1 : if (fp == nullptr)
739 : {
740 0 : CPLError(CE_Failure, CPLE_OpenFailed,
741 : "Attempt to create file `%s' failed.\n", pszFilename);
742 0 : return nullptr;
743 : }
744 :
745 : /* -------------------------------------------------------------------- */
746 : /* Write an empty header. */
747 : /* -------------------------------------------------------------------- */
748 :
749 1 : GByte abyBuf[BYN_HDR_SZ] = {'\0'};
750 :
751 : /* Load header with some commum values */
752 :
753 1 : BYNHeader hHeader = {0, 0, 0, 0, 0, 0, 0, 0, 0.0, 0, 0, 0,
754 : 0, 0, 0, 0, 0, 0.0, 0.0, 0, 0, 0.0, 0};
755 :
756 : /* Set temporary values on header */
757 :
758 1 : hHeader.nSouth = 0;
759 1 : hHeader.nNorth = nYSize - 2;
760 1 : hHeader.nWest = 0;
761 1 : hHeader.nEast = nXSize - 2;
762 1 : hHeader.nDLat = 1;
763 1 : hHeader.nDLon = 1;
764 1 : hHeader.nSizeOf = static_cast<GInt16>(GDALGetDataTypeSizeBytes(eType));
765 :
766 : /* Prepare buffer for writing */
767 :
768 1 : header2buffer(&hHeader, abyBuf);
769 :
770 : /* Write initial header */
771 :
772 1 : VSIFWriteL(abyBuf, BYN_HDR_SZ, 1, fp);
773 1 : VSIFCloseL(fp);
774 :
775 1 : return GDALDataset::FromHandle(GDALOpen(pszFilename, GA_Update));
776 : }
777 :
778 : /************************************************************************/
779 : /* UpdateHeader() */
780 : /************************************************************************/
781 :
782 1 : void BYNDataset::UpdateHeader()
783 : {
784 1 : double dfDLon = (adfGeoTransform[1] * 3600.0);
785 1 : double dfDLat = (adfGeoTransform[5] * 3600.0 * -1);
786 1 : double dfWest = (adfGeoTransform[0] * 3600.0) + (dfDLon / 2);
787 1 : double dfNorth = (adfGeoTransform[3] * 3600.0) - (dfDLat / 2);
788 1 : double dfSouth = dfNorth - ((nRasterYSize - 1) * dfDLat);
789 1 : double dfEast = dfWest + ((nRasterXSize - 1) * dfDLon);
790 :
791 1 : if (hHeader.nScale == 1)
792 : {
793 0 : dfSouth /= BYN_SCALE;
794 0 : dfNorth /= BYN_SCALE;
795 0 : dfWest /= BYN_SCALE;
796 0 : dfEast /= BYN_SCALE;
797 0 : dfDLat /= BYN_SCALE;
798 0 : dfDLon /= BYN_SCALE;
799 : }
800 :
801 1 : hHeader.nSouth = static_cast<GInt32>(dfSouth);
802 1 : hHeader.nNorth = static_cast<GInt32>(dfNorth);
803 1 : hHeader.nWest = static_cast<GInt32>(dfWest);
804 1 : hHeader.nEast = static_cast<GInt32>(dfEast);
805 1 : hHeader.nDLat = static_cast<GInt16>(dfDLat);
806 1 : hHeader.nDLon = static_cast<GInt16>(dfDLon);
807 :
808 : GByte abyBuf[BYN_HDR_SZ];
809 :
810 1 : header2buffer(&hHeader, abyBuf);
811 :
812 1 : const char *pszValue = GetMetadataItem("GLOBAL");
813 1 : if (pszValue != nullptr)
814 0 : hHeader.nGlobal = static_cast<GInt16>(atoi(pszValue));
815 :
816 1 : pszValue = GetMetadataItem("TYPE");
817 1 : if (pszValue != nullptr)
818 0 : hHeader.nType = static_cast<GInt16>(atoi(pszValue));
819 :
820 1 : pszValue = GetMetadataItem("DESCRIPTION");
821 1 : if (pszValue != nullptr)
822 0 : hHeader.nDescrip = static_cast<GInt16>(atoi(pszValue));
823 :
824 1 : pszValue = GetMetadataItem("SUBTYPE");
825 1 : if (pszValue != nullptr)
826 0 : hHeader.nSubType = static_cast<GInt16>(atoi(pszValue));
827 :
828 1 : pszValue = GetMetadataItem("WO");
829 1 : if (pszValue != nullptr)
830 0 : hHeader.dfWo = CPLAtof(pszValue);
831 :
832 1 : pszValue = GetMetadataItem("GM");
833 1 : if (pszValue != nullptr)
834 0 : hHeader.dfGM = CPLAtof(pszValue);
835 :
836 1 : pszValue = GetMetadataItem("TIDESYSTEM");
837 1 : if (pszValue != nullptr)
838 0 : hHeader.nTideSys = static_cast<GInt16>(atoi(pszValue));
839 :
840 1 : pszValue = GetMetadataItem("REALIZATION");
841 1 : if (pszValue != nullptr)
842 0 : hHeader.nRealiz = static_cast<GInt16>(atoi(pszValue));
843 :
844 1 : pszValue = GetMetadataItem("EPOCH");
845 1 : if (pszValue != nullptr)
846 0 : hHeader.dEpoch = static_cast<float>(CPLAtof(pszValue));
847 :
848 1 : pszValue = GetMetadataItem("PTTYPE");
849 1 : if (pszValue != nullptr)
850 0 : hHeader.nPtType = static_cast<GInt16>(atoi(pszValue));
851 :
852 1 : CPL_IGNORE_RET_VAL(VSIFSeekL(fpImage, 0, SEEK_SET));
853 1 : CPL_IGNORE_RET_VAL(VSIFWriteL(abyBuf, BYN_HDR_SZ, 1, fpImage));
854 :
855 : /* GDALPam metadata update */
856 :
857 1 : SetMetadataItem("GLOBAL", CPLSPrintf("%d", hHeader.nGlobal), "BYN");
858 1 : SetMetadataItem("TYPE", CPLSPrintf("%d", hHeader.nType), "BYN");
859 1 : SetMetadataItem("DESCRIPTION", CPLSPrintf("%d", hHeader.nDescrip), "BYN");
860 1 : SetMetadataItem("SUBTYPE", CPLSPrintf("%d", hHeader.nSubType), "BYN");
861 1 : SetMetadataItem("WO", CPLSPrintf("%g", hHeader.dfWo), "BYN");
862 1 : SetMetadataItem("GM", CPLSPrintf("%g", hHeader.dfGM), "BYN");
863 1 : SetMetadataItem("TIDESYSTEM", CPLSPrintf("%d", hHeader.nTideSys), "BYN");
864 1 : SetMetadataItem("REALIZATION", CPLSPrintf("%d", hHeader.nRealiz), "BYN");
865 1 : SetMetadataItem("EPOCH", CPLSPrintf("%g", hHeader.dEpoch), "BYN");
866 1 : SetMetadataItem("PTTYPE", CPLSPrintf("%d", hHeader.nPtType), "BYN");
867 1 : }
868 :
869 : /************************************************************************/
870 : /* GDALRegister_BYN() */
871 : /************************************************************************/
872 :
873 1682 : void GDALRegister_BYN()
874 :
875 : {
876 1682 : if (GDALGetDriverByName("BYN") != nullptr)
877 301 : return;
878 :
879 1381 : GDALDriver *poDriver = new GDALDriver();
880 :
881 1381 : poDriver->SetDescription("BYN");
882 1381 : poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
883 1381 : poDriver->SetMetadataItem(GDAL_DMD_LONGNAME,
884 1381 : "Natural Resources Canada's Geoid");
885 1381 : poDriver->SetMetadataItem(GDAL_DMD_EXTENSIONS, "byn err");
886 1381 : poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
887 1381 : poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/byn.html");
888 1381 : poDriver->SetMetadataItem(GDAL_DMD_CREATIONDATATYPES, "Int16 Int32");
889 :
890 1381 : poDriver->pfnOpen = BYNDataset::Open;
891 1381 : poDriver->pfnIdentify = BYNDataset::Identify;
892 1381 : poDriver->pfnCreate = BYNDataset::Create;
893 :
894 1381 : GetGDALDriverManager()->RegisterDriver(poDriver);
895 : }
|