LCOV - code coverage report
Current view: top level - frmts/netcdf - netcdfsg.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 427 489 87.3 %
Date: 2025-01-18 12:42:00 Functions: 19 23 82.6 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  netCDF read/write Driver
       4             :  * Purpose:  GDAL bindings over netCDF library.
       5             :  * Author:   Winor Chen <wchen329 at wisc.edu>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2019, Winor Chen <wchen329 at wisc.edu>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : #include <cstdio>
      13             : #include <cstring>
      14             : #include <set>
      15             : #include <vector>
      16             : #include "netcdf.h"
      17             : #include "netcdfdataset.h"
      18             : #include "netcdfsg.h"
      19             : 
      20             : namespace nccfdriver
      21             : {
      22             : /* Attribute Fetch
      23             :  * -
      24             :  * A function which makes it a bit easier to fetch single text attribute values
      25             :  * ncid: as used in netcdf.h
      26             :  * varID: variable id in which to look for the attribute
      27             :  * attrName: name of attribute to fine
      28             :  * alloc: a reference to a string that will be filled with the attribute (i.e.
      29             :  * truncated and filled with the return value) Returns: a reference to the
      30             :  * string to fill (a.k.a. string pointed to by alloc reference)
      31             :  */
      32        1072 : std::string &attrf(int ncid, int varId, const char *attrName,
      33             :                    std::string &alloc)
      34             : {
      35        1072 :     size_t len = 0;
      36        1072 :     nc_inq_attlen(ncid, varId, attrName, &len);
      37             : 
      38        1072 :     if (len < 1)
      39             :     {
      40         188 :         alloc.clear();
      41         188 :         return alloc;
      42             :     }
      43             : 
      44         884 :     alloc.resize(len);
      45         884 :     memset(&alloc[0], 0, len);
      46             : 
      47             :     // Now look through this variable for the attribute
      48         884 :     nc_get_att_text(ncid, varId, attrName, &alloc[0]);
      49             : 
      50         884 :     return alloc;
      51             : }
      52             : 
      53             : /* SGeometry_Reader
      54             :  * (implementations)
      55             :  *
      56             :  */
      57          73 : SGeometry_Reader::SGeometry_Reader(int ncId, int geoVarId)
      58         139 :     : gc_varId(geoVarId), touple_order(0)
      59             : {
      60             : 
      61             :     char container_name[NC_MAX_NAME + 1];
      62          73 :     memset(container_name, 0, NC_MAX_NAME + 1);
      63             : 
      64             :     // Get geometry container name
      65          73 :     if (nc_inq_varname(ncId, geoVarId, container_name) != NC_NOERR)
      66             :     {
      67             :         throw SG_Exception_Existential("new geometry container",
      68           0 :                                        "the variable of the given ID");
      69             :     }
      70             : 
      71             :     // Establish string version of container_name
      72          73 :     container_name_s = std::string(container_name);
      73             : 
      74             :     // Find geometry type
      75          73 :     this->type = nccfdriver::getGeometryType(ncId, geoVarId);
      76             : 
      77          73 :     if (this->type == NONE)
      78             :     {
      79             :         throw SG_Exception_Existential(
      80           1 :             static_cast<const char *>(container_name), CF_SG_GEOMETRY_TYPE);
      81             :     }
      82             : 
      83             :     // Get grid mapping variable, if it exists
      84          72 :     this->gm_varId = INVALID_VAR_ID;
      85          72 :     if (attrf(ncId, geoVarId, CF_GRD_MAPPING, gm_name_s) != "")
      86             :     {
      87          46 :         const char *gm_name = gm_name_s.c_str();
      88             :         int gmVID;
      89          46 :         if (nc_inq_varid(ncId, gm_name, &gmVID) == NC_NOERR)
      90             :         {
      91          46 :             this->gm_varId = gmVID;
      92             :         }
      93             :     }
      94             : 
      95             :     // Find a list of node counts and part node count
      96         144 :     std::string nc_name_s;
      97         144 :     std::string pnc_name_s;
      98         144 :     std::string ir_name_s;
      99          72 :     int pnc_vid = INVALID_VAR_ID;
     100          72 :     int nc_vid = INVALID_VAR_ID;
     101          72 :     int ir_vid = INVALID_VAR_ID;
     102             :     int buf;
     103          72 :     size_t bound = 0;
     104          72 :     size_t total_node_count = 0;  // used in error checks later
     105          72 :     size_t total_part_node_count = 0;
     106          72 :     if (attrf(ncId, geoVarId, CF_SG_NODE_COUNT, nc_name_s) != "")
     107             :     {
     108          58 :         const char *nc_name = nc_name_s.c_str();
     109          58 :         nc_inq_varid(ncId, nc_name, &nc_vid);
     110         651 :         while (nc_get_var1_int(ncId, nc_vid, &bound, &buf) == NC_NOERR)
     111             :         {
     112         594 :             if (buf < 0)
     113             :             {
     114             :                 throw SG_Exception_Value_Violation(
     115             :                     static_cast<const char *>(container_name), CF_SG_NODE_COUNT,
     116           1 :                     "negative");
     117             :             }
     118             : 
     119         593 :             this->node_counts.push_back(buf);
     120         593 :             total_node_count += buf;
     121         593 :             bound++;
     122             :         }
     123             :     }
     124             : 
     125          71 :     if (attrf(ncId, geoVarId, CF_SG_PART_NODE_COUNT, pnc_name_s) != "")
     126             :     {
     127          38 :         const char *pnc_name = pnc_name_s.c_str();
     128          38 :         bound = 0;
     129          38 :         nc_inq_varid(ncId, pnc_name, &pnc_vid);
     130         870 :         while (nc_get_var1_int(ncId, pnc_vid, &bound, &buf) == NC_NOERR)
     131             :         {
     132         833 :             if (buf < 0)
     133             :             {
     134             :                 throw SG_Exception_Value_Violation(
     135             :                     static_cast<const char *>(container_name),
     136           1 :                     CF_SG_PART_NODE_COUNT, "negative");
     137             :             }
     138             : 
     139         832 :             this->pnode_counts.push_back(buf);
     140         832 :             total_part_node_count += buf;
     141         832 :             bound++;
     142             :         }
     143             :     }
     144             : 
     145          70 :     if (attrf(ncId, geoVarId, CF_SG_INTERIOR_RING, ir_name_s) != "")
     146             :     {
     147          12 :         const char *ir_name = ir_name_s.c_str();
     148          12 :         bound = 0;
     149          12 :         nc_inq_varid(ncId, ir_name, &ir_vid);
     150         295 :         while (nc_get_var1_int(ncId, ir_vid, &bound, &buf) == NC_NOERR)
     151             :         {
     152         284 :             bool store = buf == 0 ? false : true;
     153             : 
     154         284 :             if (buf != 0 && buf != 1)
     155             :             {
     156             :                 throw SG_Exception_Value_Required(
     157             :                     static_cast<const char *>(container_name),
     158           1 :                     CF_SG_INTERIOR_RING, "0 or 1");
     159             :             }
     160             : 
     161         283 :             this->int_rings.push_back(store);
     162         283 :             bound++;
     163             :         }
     164             :     }
     165             : 
     166             :     /* Enforcement of well formed CF files
     167             :      * If these are not met then the dataset is malformed and will halt further
     168             :      * processing of simple geometries.
     169             :      */
     170             : 
     171             :     // part node count exists only when node count exists
     172          69 :     if (pnode_counts.size() > 0 && node_counts.size() == 0)
     173             :     {
     174             :         throw SG_Exception_Dep(static_cast<const char *>(container_name),
     175           0 :                                CF_SG_PART_NODE_COUNT, CF_SG_NODE_COUNT);
     176             :     }
     177             : 
     178             :     // part node counts and node counts don't match up in count
     179          69 :     if (pnc_vid != INVALID_VAR_ID && total_node_count != total_part_node_count)
     180             :     {
     181             :         throw SG_Exception_BadSum(static_cast<const char *>(container_name),
     182           0 :                                   CF_SG_PART_NODE_COUNT, CF_SG_PART_NODE_COUNT);
     183             :     }
     184             : 
     185             :     // interior rings only exist when part node counts exist
     186          69 :     if (int_rings.size() > 0 && pnode_counts.size() == 0)
     187             :     {
     188             :         throw SG_Exception_Dep(static_cast<const char *>(container_name),
     189           0 :                                CF_SG_INTERIOR_RING, CF_SG_PART_NODE_COUNT);
     190             :     }
     191             : 
     192             :     // cardinality of part_node_counts == cardinality of interior_ring (if
     193             :     // interior ring > 0)
     194          69 :     if (int_rings.size() > 0)
     195             :     {
     196          11 :         if (int_rings.size() != pnode_counts.size())
     197             :         {
     198             :             throw SG_Exception_Dim_MM(static_cast<const char *>(container_name),
     199             :                                       CF_SG_INTERIOR_RING,
     200           0 :                                       CF_SG_PART_NODE_COUNT);
     201             :         }
     202             :     }
     203             : 
     204             :     // lines and polygons require node_counts, multipolygons are checked with
     205             :     // part_node_count
     206          69 :     if (this->type == POLYGON || this->type == LINE)
     207             :     {
     208          14 :         if (node_counts.size() < 1)
     209             :         {
     210             :             throw SG_Exception_Existential(
     211           1 :                 static_cast<const char *>(container_name), CF_SG_NODE_COUNT);
     212             :         }
     213             :     }
     214             : 
     215             :     /* Basic Safety checks End
     216             :      */
     217             : 
     218             :     // Create bound list
     219          68 :     size_t rc = 0;
     220          68 :     bound_list.push_back(0);  // start with 0
     221             : 
     222          68 :     if (node_counts.size() > 0)
     223             :     {
     224         587 :         for (size_t i = 0; i < node_counts.size() - 1; i++)
     225             :         {
     226         532 :             rc = rc + node_counts[i];
     227         532 :             bound_list.push_back(rc);
     228             :         }
     229             :     }
     230             : 
     231          69 :     std::string cart_s;
     232             :     // Node Coordinates
     233          68 :     if (attrf(ncId, geoVarId, CF_SG_NODE_COORDINATES, cart_s) == "")
     234             :     {
     235           0 :         throw SG_Exception_Existential(container_name, CF_SG_NODE_COORDINATES);
     236             :     }
     237             : 
     238             :     // Create parts count list and an offset list for parts indexing
     239          68 :     if (this->node_counts.size() > 0)
     240             :     {
     241          55 :         int ind = 0;
     242          55 :         int parts = 0;
     243          55 :         int prog = 0;
     244          55 :         int c = 0;
     245             : 
     246         883 :         for (size_t pcnt = 0; pcnt < pnode_counts.size(); pcnt++)
     247             :         {
     248         828 :             if (prog == 0)
     249         521 :                 pnc_bl.push_back(pcnt);
     250             : 
     251         828 :             if (int_rings.size() > 0 && !int_rings[pcnt])
     252         236 :                 c++;
     253             : 
     254         828 :             prog = prog + pnode_counts[pcnt];
     255         828 :             parts++;
     256             : 
     257         828 :             if (prog == node_counts[ind])
     258             :             {
     259         521 :                 ind++;
     260         521 :                 this->parts_count.push_back(parts);
     261         521 :                 if (int_rings.size() > 0)
     262         230 :                     this->poly_count.push_back(c);
     263         521 :                 c = 0;
     264         521 :                 prog = 0;
     265         521 :                 parts = 0;
     266             :             }
     267         307 :             else if (prog > node_counts[ind])
     268             :             {
     269             :                 throw SG_Exception_BadSum(container_name, CF_SG_PART_NODE_COUNT,
     270           0 :                                           CF_SG_NODE_COUNT);
     271             :             }
     272             :         }
     273             :     }
     274             : 
     275             :     // (1) the tuple order for a single point
     276             :     // (2) the variable ids with the relevant coordinate values
     277          68 :     int X = INVALID_VAR_ID;
     278          68 :     int Y = INVALID_VAR_ID;
     279          68 :     int Z = INVALID_VAR_ID;
     280             : 
     281             :     char cart[NC_MAX_NAME + 1];
     282          68 :     memset(cart, 0, NC_MAX_NAME + 1);
     283          68 :     strncpy(cart, cart_s.c_str(), NC_MAX_NAME);
     284             : 
     285          68 :     char *dim = strtok(cart, " ");
     286          68 :     int axis_id = 0;
     287             : 
     288         228 :     while (dim != nullptr)
     289             :     {
     290         160 :         if (nc_inq_varid(ncId, dim, &axis_id) == NC_NOERR)
     291             :         {
     292             : 
     293             :             // Check axis signature
     294         160 :             std::string a_sig;
     295         160 :             attrf(ncId, axis_id, CF_AXIS, a_sig);
     296             : 
     297             :             // If valid signify axis correctly
     298         160 :             if (a_sig == "X")
     299             :             {
     300          68 :                 X = axis_id;
     301             :             }
     302          92 :             else if (a_sig == "Y")
     303             :             {
     304          68 :                 Y = axis_id;
     305             :             }
     306          24 :             else if (a_sig == "Z")
     307             :             {
     308          24 :                 Z = axis_id;
     309             :             }
     310             :             else
     311             :             {
     312             :                 throw SG_Exception_Dep(container_name,
     313           0 :                                        "A node_coordinates variable", CF_AXIS);
     314             :             }
     315             : 
     316         160 :             this->touple_order++;
     317             :         }
     318             :         else
     319             :         {
     320           0 :             throw SG_Exception_Existential(container_name, dim);
     321             :         }
     322             : 
     323         160 :         dim = strtok(nullptr, " ");
     324             :     }
     325             : 
     326             :     // Write axis in X, Y, Z order
     327             : 
     328          68 :     if (X != INVALID_VAR_ID)
     329          68 :         this->nodec_varIds.push_back(X);
     330             :     else
     331             :     {
     332             :         throw SG_Exception_Existential(container_name,
     333           0 :                                        "node_coordinates: X-axis");
     334             :     }
     335          68 :     if (Y != INVALID_VAR_ID)
     336          68 :         this->nodec_varIds.push_back(Y);
     337             :     else
     338             :     {
     339             :         throw SG_Exception_Existential(container_name,
     340           0 :                                        "node_coordinates: Y-axis");
     341             :     }
     342          68 :     if (Z != INVALID_VAR_ID)
     343          24 :         this->nodec_varIds.push_back(Z);
     344             : 
     345             :     /* Final Checks for node coordinates
     346             :      * (1) Each axis has one and only one dimension, dim length of each node
     347             :      * coordinates are all equal (2) total node count == dim length of each node
     348             :      * coordinates (if node_counts not empty) (3) there are at least two node
     349             :      * coordinate variable ids
     350             :      */
     351             : 
     352          68 :     int all_dim = INVALID_VAR_ID;
     353          68 :     bool dim_set = false;
     354             :     //(1) one dimension check, each node_coordinates have same dimension
     355         227 :     for (size_t nvitr = 0; nvitr < nodec_varIds.size(); nvitr++)
     356             :     {
     357         160 :         int dimC = 0;
     358         160 :         nc_inq_varndims(ncId, nodec_varIds[nvitr], &dimC);
     359             : 
     360         160 :         if (dimC != 1)
     361             :         {
     362           0 :             throw SG_Exception_Not1D();
     363             :         }
     364             : 
     365             :         // check that all have same dimension
     366             :         int inter_dim[1];
     367         160 :         if (nc_inq_vardimid(ncId, nodec_varIds[nvitr], inter_dim) != NC_NOERR)
     368             :         {
     369             :             throw SG_Exception_Existential(
     370           0 :                 container_name, "one or more node_coordinate dimensions");
     371             :         }
     372             : 
     373         160 :         if (!dim_set)
     374             :         {
     375          68 :             all_dim = inter_dim[0];
     376          68 :             dim_set = true;
     377             :         }
     378             : 
     379             :         else
     380             :         {
     381          92 :             if (inter_dim[0] != all_dim)
     382             :                 throw SG_Exception_Dim_MM(
     383             :                     container_name, "X, Y",
     384           1 :                     "in general all node coordinate axes");
     385             :         }
     386             :     }
     387             : 
     388             :     // (2) check equality one
     389          67 :     if (node_counts.size() > 0)
     390             :     {
     391          54 :         size_t diml = 0;
     392          54 :         nc_inq_dimlen(ncId, all_dim, &diml);
     393             : 
     394          54 :         if (diml != total_node_count)
     395             :             throw SG_Exception_BadSum(container_name, "node_count",
     396           0 :                                       "node coordinate dimension length");
     397             :     }
     398             : 
     399             :     // (3) check tuple order
     400          67 :     if (this->touple_order < 2)
     401             :     {
     402             :         throw SG_Exception_Existential(
     403             :             container_name,
     404           0 :             "insufficient node coordinates must have at least two axis");
     405             :     }
     406             : 
     407             :     /* Investigate for instance dimension
     408             :      * The procedure is as follows
     409             :      *
     410             :      * (1) if there's node_count, use the dimension used to index node count
     411             :      * (2) otherwise it's point (singleton) data, in this case use the node
     412             :      * coordinate dimension
     413             :      */
     414          67 :     size_t instance_dim_len = 0;
     415             : 
     416          67 :     if (node_counts.size() >= 1)
     417             :     {
     418          54 :         int nc_dims = 0;
     419          54 :         nc_inq_varndims(ncId, nc_vid, &nc_dims);
     420             : 
     421          54 :         if (nc_dims != 1)
     422           0 :             throw SG_Exception_Not1D();
     423             : 
     424             :         int nc_dim_id[1];
     425             : 
     426          54 :         if (nc_inq_vardimid(ncId, nc_vid, nc_dim_id) != NC_NOERR)
     427             :         {
     428             :             throw SG_Exception_Existential(container_name,
     429           0 :                                            "node_count dimension");
     430             :         }
     431             : 
     432          54 :         this->inst_dimId = nc_dim_id[0];
     433             :     }
     434             : 
     435             :     else
     436             :     {
     437          13 :         this->inst_dimId = all_dim;
     438             :     }
     439             : 
     440          67 :     nc_inq_dimlen(ncId, this->inst_dimId, &instance_dim_len);
     441             : 
     442          67 :     if (instance_dim_len == 0)
     443           0 :         throw SG_Exception_EmptyDim();
     444             : 
     445             :     // Set values accordingly
     446          67 :     this->inst_dimLen = instance_dim_len;
     447          67 :     this->pt_buffer = std::make_unique<Point>(this->touple_order);
     448          67 :     this->gc_varId = geoVarId;
     449          67 :     this->ncid = ncId;
     450          67 : }
     451             : 
     452       67341 : Point &SGeometry_Reader::operator[](size_t index)
     453             : {
     454      202262 :     for (int order = 0; order < touple_order; order++)
     455             :     {
     456      134921 :         Point &pt = *(this->pt_buffer);
     457             :         double data;
     458      134921 :         size_t real_ind = index;
     459             : 
     460             :         // Read a single coord
     461             :         int err =
     462      134921 :             nc_get_var1_double(ncid, nodec_varIds[order], &real_ind, &data);
     463             : 
     464      134921 :         if (err != NC_NOERR)
     465             :         {
     466           0 :             throw SG_Exception_BadPoint();
     467             :         }
     468             : 
     469      134921 :         pt[order] = data;
     470             :     }
     471             : 
     472       67341 :     return *(this->pt_buffer);
     473             : }
     474             : 
     475         529 : size_t SGeometry_Reader::get_geometry_count()
     476             : {
     477         529 :     if (type == POINT)
     478             :     {
     479          47 :         if (this->nodec_varIds.size() < 1)
     480           0 :             return 0;
     481             : 
     482             :         // If more than one dim, then error. Otherwise inquire its length and
     483             :         // return that
     484             :         int dims;
     485          47 :         if (nc_inq_varndims(this->ncid, nodec_varIds[0], &dims) != NC_NOERR)
     486           0 :             return 0;
     487          47 :         if (dims != 1)
     488           0 :             return 0;
     489             : 
     490             :         // Find which dimension is used for x
     491             :         int index;
     492          47 :         if (nc_inq_vardimid(this->ncid, nodec_varIds[0], &index) != NC_NOERR)
     493             :         {
     494           0 :             return 0;
     495             :         }
     496             : 
     497             :         // Finally find the length
     498             :         size_t len;
     499          47 :         if (nc_inq_dimlen(this->ncid, index, &len) != NC_NOERR)
     500             :         {
     501           0 :             return 0;
     502             :         }
     503          47 :         return len;
     504             :     }
     505             : 
     506             :     else
     507         482 :         return this->node_counts.size();
     508             : }
     509             : 
     510        1200 : static void add_to_buffer(std::vector<unsigned char> &buffer, uint8_t v)
     511             : {
     512        1200 :     buffer.push_back(v);
     513        1200 : }
     514             : 
     515        2899 : static void add_to_buffer(std::vector<unsigned char> &buffer, uint32_t v)
     516             : {
     517        2899 :     const size_t old_size = buffer.size();
     518        2899 :     buffer.resize(old_size + sizeof(v));
     519        2899 :     memcpy(&buffer[old_size], &v, sizeof(v));
     520        2899 : }
     521             : 
     522             : static void add_to_buffer(std::vector<unsigned char> &, int) = delete;
     523             : 
     524      134921 : static void add_to_buffer(std::vector<unsigned char> &buffer, double v)
     525             : {
     526      134921 :     const size_t old_size = buffer.size();
     527      134921 :     buffer.resize(old_size + sizeof(v));
     528      134921 :     memcpy(&buffer[old_size], &v, sizeof(v));
     529      134921 : }
     530             : 
     531             : /* serializeToWKB
     532             :  * Takes the geometry in SGeometry at a given index and converts it into WKB
     533             :  * format. Converting SGeometry into WKB automatically allocates the required
     534             :  * buffer space and returns a buffer that MUST be free'd
     535             :  */
     536         492 : std::vector<unsigned char> SGeometry_Reader::serializeToWKB(size_t featureInd)
     537             : {
     538         492 :     std::vector<unsigned char> ret;
     539         492 :     int nc = 0;
     540         492 :     size_t sb = 0;
     541             : 
     542             :     // Points don't have node_count entry... only inspect and set node_counts if
     543             :     // not a point
     544         492 :     if (this->getGeometryType() != POINT)
     545             :     {
     546         453 :         nc = node_counts[featureInd];
     547         453 :         sb = bound_list[featureInd];
     548             :     }
     549             : 
     550             :     // Serialization occurs differently depending on geometry
     551             :     // The memory requirements also differ between geometries
     552         492 :     switch (this->getGeometryType())
     553             :     {
     554          39 :         case POINT:
     555          39 :             inPlaceSerialize_Point(this, featureInd, ret);
     556          39 :             break;
     557             : 
     558          16 :         case LINE:
     559          16 :             inPlaceSerialize_LineString(this, nc, sb, ret);
     560          16 :             break;
     561             : 
     562          13 :         case POLYGON:
     563             :             // A polygon has:
     564             :             // 1 byte header
     565             :             // 4 byte Type
     566             :             // 4 byte ring count (1 (exterior)) [assume only exterior rings,
     567             :             // otherwise multipolygon] For each ring: 4 byte point count, 8 byte
     568             :             // double x point count x # dimension (This is equivalent to
     569             :             // requesting enough for all points and a point count header for
     570             :             // each point) (Or 8 byte double x node count x # dimension + 4 byte
     571             :             // point count x part_count)
     572             : 
     573             :             // if interior ring, then assume that it will be a multipolygon
     574             :             // (maybe future work?)
     575          13 :             inPlaceSerialize_PolygonExtOnly(this, nc, sb, ret);
     576          13 :             break;
     577             : 
     578          25 :         case MULTIPOINT:
     579             :         {
     580          25 :             uint8_t header = PLATFORM_HEADER;
     581          32 :             uint32_t t = this->touple_order == 2   ? wkbMultiPoint
     582           7 :                          : this->touple_order == 3 ? wkbMultiPoint25D
     583             :                                                    : wkbNone;
     584             : 
     585          25 :             if (t == wkbNone)
     586           0 :                 throw SG_Exception_BadFeature();
     587             : 
     588             :             // Add metadata
     589          25 :             add_to_buffer(ret, header);
     590          25 :             add_to_buffer(ret, t);
     591          25 :             add_to_buffer(ret, static_cast<uint32_t>(nc));
     592             : 
     593             :             // Add points
     594         119 :             for (int pts = 0; pts < nc; pts++)
     595             :             {
     596          94 :                 inPlaceSerialize_Point(this, static_cast<size_t>(sb + pts),
     597             :                                        ret);
     598             :             }
     599             :         }
     600             : 
     601          25 :         break;
     602             : 
     603          18 :         case MULTILINE:
     604             :         {
     605          18 :             uint8_t header = PLATFORM_HEADER;
     606          25 :             uint32_t t = this->touple_order == 2   ? wkbMultiLineString
     607           7 :                          : this->touple_order == 3 ? wkbMultiLineString25D
     608             :                                                    : wkbNone;
     609             : 
     610          18 :             if (t == wkbNone)
     611           0 :                 throw SG_Exception_BadFeature();
     612          18 :             int32_t lc = parts_count[featureInd];
     613          18 :             size_t seek_begin = sb;
     614             :             size_t pc_begin =
     615          18 :                 pnc_bl[featureInd];  // initialize with first part count, list
     616             :                                      // of part counts is contiguous
     617          36 :             std::vector<int> pnc;
     618             : 
     619             :             // Build sub vector for part_node_counts
     620             :             // + Calculate wkbSize
     621          47 :             for (int itr = 0; itr < lc; itr++)
     622             :             {
     623          29 :                 pnc.push_back(pnode_counts[pc_begin + itr]);
     624             :             }
     625             : 
     626          18 :             size_t cur_point = seek_begin;
     627          18 :             size_t pcount = pnc.size();
     628             : 
     629             :             // Begin Writing
     630          18 :             add_to_buffer(ret, header);
     631          18 :             add_to_buffer(ret, t);
     632          18 :             add_to_buffer(ret, static_cast<uint32_t>(pcount));
     633             : 
     634          47 :             for (size_t itr = 0; itr < pcount; itr++)
     635             :             {
     636          29 :                 inPlaceSerialize_LineString(this, pnc[itr], cur_point, ret);
     637          29 :                 cur_point = pnc[itr] + cur_point;
     638             :             }
     639             :         }
     640             : 
     641          18 :         break;
     642             : 
     643         381 :         case MULTIPOLYGON:
     644             :         {
     645         381 :             uint8_t header = PLATFORM_HEADER;
     646         395 :             uint32_t t = this->touple_order == 2   ? wkbMultiPolygon
     647          14 :                          : this->touple_order == 3 ? wkbMultiPolygon25D
     648             :                                                    : wkbNone;
     649             : 
     650         381 :             if (t == wkbNone)
     651           0 :                 throw SG_Exception_BadFeature();
     652         381 :             bool noInteriors = this->int_rings.size() == 0 ? true : false;
     653         381 :             int32_t rc = parts_count[featureInd];
     654         381 :             size_t seek_begin = sb;
     655             :             size_t pc_begin =
     656         381 :                 pnc_bl[featureInd];  // initialize with first part count, list
     657             :                                      // of part counts is contiguous
     658         762 :             std::vector<int> pnc;
     659             : 
     660             :             // Build sub vector for part_node_counts
     661        1000 :             for (int itr = 0; itr < rc; itr++)
     662             :             {
     663         619 :                 pnc.push_back(pnode_counts[pc_begin + itr]);
     664             :             }
     665             : 
     666             :             // Create Multipolygon headers
     667         381 :             add_to_buffer(ret, header);
     668         381 :             add_to_buffer(ret, t);
     669             : 
     670         381 :             if (noInteriors)
     671             :             {
     672         221 :                 size_t cur_point = seek_begin;
     673         221 :                 size_t pcount = pnc.size();
     674         221 :                 add_to_buffer(ret, static_cast<uint32_t>(pcount));
     675             : 
     676         640 :                 for (size_t itr = 0; itr < pcount; itr++)
     677             :                 {
     678         419 :                     inPlaceSerialize_PolygonExtOnly(this, pnc[itr], cur_point,
     679             :                                                     ret);
     680         419 :                     cur_point = pnc[itr] + cur_point;
     681             :                 }
     682             :             }
     683             : 
     684             :             else
     685             :             {
     686         160 :                 int32_t polys = poly_count[featureInd];
     687         160 :                 add_to_buffer(ret, static_cast<uint32_t>(polys));
     688             : 
     689         160 :                 size_t base = pnc_bl[featureInd];  // beginning of parts_count
     690             :                                                    // for this multigeometry
     691         160 :                 size_t seek = seek_begin;  // beginning of node range for this
     692             :                                            // multigeometry
     693         160 :                 size_t ir_base = base + 1;
     694             : 
     695             :                 // has interior rings,
     696         326 :                 for (int32_t itr = 0; itr < polys; itr++)
     697             :                 {
     698         166 :                     int rc_m = 1;
     699             : 
     700             :                     // count how many parts belong to each Polygon
     701         200 :                     while (ir_base < int_rings.size() && int_rings[ir_base])
     702             :                     {
     703          34 :                         rc_m++;
     704          34 :                         ir_base++;
     705             :                     }
     706             : 
     707         166 :                     if (rc_m == 1)
     708         142 :                         ir_base++;  // single ring case
     709             : 
     710         166 :                     std::vector<int> poly_parts;
     711             : 
     712             :                     // Make part node count sub vector
     713         366 :                     for (int itr_2 = 0; itr_2 < rc_m; itr_2++)
     714             :                     {
     715         200 :                         poly_parts.push_back(pnode_counts[base + itr_2]);
     716             :                     }
     717             : 
     718         166 :                     inPlaceSerialize_Polygon(this, poly_parts, rc_m, seek, ret);
     719             : 
     720             :                     // Update seek position
     721         366 :                     for (size_t itr_3 = 0; itr_3 < poly_parts.size(); itr_3++)
     722             :                     {
     723         200 :                         seek += poly_parts[itr_3];
     724             :                     }
     725             : 
     726         166 :                     base += poly_parts.size();
     727             :                     // cppcheck-suppress redundantAssignment
     728         166 :                     ir_base = base + 1;
     729             :                 }
     730             :             }
     731             :         }
     732         381 :         break;
     733             : 
     734           0 :         default:
     735             : 
     736           0 :             throw SG_Exception_BadFeature();
     737             :             break;
     738             :     }
     739             : 
     740         492 :     return ret;
     741             : }
     742             : 
     743          67 : void SGeometry_PropertyScanner::open(int container_id)
     744             : {
     745             :     // First check for container_id, if variable doesn't exist error out
     746          67 :     if (nc_inq_var(this->nc, container_id, nullptr, nullptr, nullptr, nullptr,
     747          67 :                    nullptr) != NC_NOERR)
     748             :     {
     749           0 :         return;  // change to exception
     750             :     }
     751             : 
     752             :     // Now exists, see what variables refer to this one
     753             :     // First get name of this container
     754             :     char contname[NC_MAX_NAME + 1];
     755          67 :     memset(contname, 0, NC_MAX_NAME + 1);
     756          67 :     if (nc_inq_varname(this->nc, container_id, contname) != NC_NOERR)
     757             :     {
     758           0 :         return;
     759             :     }
     760             : 
     761             :     // Then scan throughout the netcdfDataset if those variables
     762             :     // geometry_container attribute matches the container
     763          67 :     int varCount = 0;
     764          67 :     if (nc_inq_nvars(this->nc, &varCount) != NC_NOERR)
     765             :     {
     766           0 :         return;
     767             :     }
     768             : 
     769         615 :     for (int curr = 0; curr < varCount; curr++)
     770             :     {
     771         548 :         size_t contname2_len = 0;
     772             : 
     773             :         // First find container length, and make buf that size in chars
     774         548 :         if (nc_inq_attlen(this->nc, curr, CF_SG_GEOMETRY, &contname2_len) !=
     775             :             NC_NOERR)
     776             :         {
     777             :             // not a geometry variable, continue
     778         427 :             continue;
     779             :         }
     780             : 
     781             :         // Also if present but empty, go on
     782         121 :         if (contname2_len == 0)
     783           0 :             continue;
     784             : 
     785             :         // Otherwise, geometry: see what container it has
     786             :         char buf[NC_MAX_CHAR + 1];
     787         121 :         memset(buf, 0, NC_MAX_CHAR + 1);
     788             : 
     789         121 :         if (nc_get_att_text(this->nc, curr, CF_SG_GEOMETRY, buf) != NC_NOERR)
     790             :         {
     791           0 :             continue;
     792             :         }
     793             : 
     794             :         // If matches, then establish a reference by placing this variable's
     795             :         // data in both vectors
     796         121 :         if (!strcmp(contname, buf))
     797             :         {
     798         109 :             char property_name[NC_MAX_NAME + 1] = {0};
     799             : 
     800             :             // look for special OGR original name field
     801         109 :             if (nc_get_att_text(this->nc, curr, OGR_SG_ORIGINAL_LAYERNAME,
     802         109 :                                 property_name) != NC_NOERR)
     803             :             {
     804             :                 // if doesn't exist, then just use the variable name
     805         109 :                 if (nc_inq_varname(this->nc, curr, property_name) != NC_NOERR)
     806             :                 {
     807           0 :                     throw SG_Exception_General_Malformed(contname);
     808             :                 }
     809             :             }
     810             : 
     811         218 :             std::string n(property_name);
     812         109 :             v_ids.push_back(curr);
     813         109 :             v_headers.push_back(n);
     814             :         }
     815             :     }
     816             : }
     817             : 
     818             : // Exception Class Implementations
     819           1 : SG_Exception_Dim_MM::SG_Exception_Dim_MM(const char *container_name,
     820             :                                          const char *field_1,
     821           1 :                                          const char *field_2)
     822             : {
     823           2 :     std::string cn_s(container_name);
     824           2 :     std::string field1_s(field_1);
     825           1 :     std::string field2_s(field_2);
     826             : 
     827           2 :     this->err_msg = "[" + cn_s + "] One or more dimensions of " + field1_s +
     828           1 :                     " and " + field2_s + " do not match but must match.";
     829           1 : }
     830             : 
     831           2 : SG_Exception_Existential::SG_Exception_Existential(const char *container_name,
     832           2 :                                                    const char *missing_name)
     833             : {
     834           4 :     std::string cn_s(container_name);
     835           2 :     std::string mn_s(missing_name);
     836             : 
     837           4 :     this->err_msg = "[" + cn_s +
     838           4 :                     "] The property or the variable associated with " + mn_s +
     839           2 :                     " is missing.";
     840           2 : }
     841             : 
     842           0 : SG_Exception_Dep::SG_Exception_Dep(const char *container_name, const char *arg1,
     843           0 :                                    const char *arg2)
     844             : {
     845           0 :     std::string cn_s(container_name);
     846           0 :     std::string arg1_s(arg1);
     847           0 :     std::string arg2_s(arg2);
     848             : 
     849           0 :     this->err_msg = "[" + cn_s + "] The attribute " + arg1_s +
     850           0 :                     " may not exist without the attribute " + arg2_s +
     851           0 :                     " existing.";
     852           0 : }
     853             : 
     854           0 : SG_Exception_BadSum::SG_Exception_BadSum(const char *container_name,
     855           0 :                                          const char *arg1, const char *arg2)
     856             : {
     857           0 :     std::string cn_s(container_name);
     858           0 :     std::string arg1_s(arg1);
     859           0 :     std::string arg2_s(arg2);
     860             : 
     861           0 :     this->err_msg = "[" + cn_s + "]" + " The sum of all values in " + arg1_s +
     862           0 :                     " and " + arg2_s + " do not match.";
     863           0 : }
     864             : 
     865           0 : SG_Exception_General_Malformed ::SG_Exception_General_Malformed(const char *arg)
     866             : {
     867           0 :     std::string arg1_s(arg);
     868             : 
     869             :     this->err_msg =
     870           0 :         "Corruption or malformed formatting has been detected in: " + arg1_s;
     871           0 : }
     872             : 
     873             : // to get past linker
     874           7 : SG_Exception::~SG_Exception()
     875             : {
     876           7 : }
     877             : 
     878             : // Helpers
     879             : // following is a short hand for a clean up and exit, since goto isn't allowed
     880         486 : double getCFVersion(int ncid)
     881             : {
     882         486 :     double ver = -1.0;
     883         972 :     std::string attrVal;
     884             : 
     885             :     // Fetch the CF attribute
     886         486 :     if (attrf(ncid, NC_GLOBAL, NCDF_CONVENTIONS, attrVal) == "")
     887             :     {
     888          56 :         return ver;
     889             :     }
     890             : 
     891         430 :     if (sscanf(attrVal.c_str(), "CF-%lf", &ver) != 1)
     892             :     {
     893          32 :         return -1.0;
     894             :     }
     895             : 
     896         398 :     return ver;
     897             : }
     898             : 
     899          73 : geom_t getGeometryType(int ncid, int varid)
     900             : {
     901          73 :     geom_t ret = UNSUPPORTED;
     902         146 :     std::string gt_name_s;
     903             :     const char *gt_name =
     904          73 :         attrf(ncid, varid, CF_SG_GEOMETRY_TYPE, gt_name_s).c_str();
     905             : 
     906          73 :     if (gt_name[0] == '\0')
     907             :     {
     908           1 :         return NONE;
     909             :     }
     910             : 
     911             :     // Points
     912          72 :     if (!strcmp(gt_name, CF_SG_TYPE_POINT))
     913             :     {
     914             :         // Node Count not present? Assume that it is a multipoint.
     915          18 :         if (nc_inq_att(ncid, varid, CF_SG_NODE_COUNT, nullptr, nullptr) ==
     916             :             NC_ENOTATT)
     917             :         {
     918          12 :             ret = POINT;
     919             :         }
     920             :         else
     921           6 :             ret = MULTIPOINT;
     922             :     }
     923             : 
     924             :     // Lines
     925          54 :     else if (!strcmp(gt_name, CF_SG_TYPE_LINE))
     926             :     {
     927             :         // Part Node Count present? Assume multiline
     928          10 :         if (nc_inq_att(ncid, varid, CF_SG_PART_NODE_COUNT, nullptr, nullptr) ==
     929             :             NC_ENOTATT)
     930             :         {
     931           4 :             ret = LINE;
     932             :         }
     933             :         else
     934           6 :             ret = MULTILINE;
     935             :     }
     936             : 
     937             :     // Polygons
     938          44 :     else if (!strcmp(gt_name, CF_SG_TYPE_POLY))
     939             :     {
     940             :         /* Polygons versus MultiPolygons, slightly ambiguous
     941             :          * Part Node Count & no Interior Ring - MultiPolygon
     942             :          * no Part Node Count & no Interior Ring - Polygon
     943             :          * Part Node Count & Interior Ring - assume that it is a MultiPolygon
     944             :          */
     945             :         int pnc_present =
     946          43 :             nc_inq_att(ncid, varid, CF_SG_PART_NODE_COUNT, nullptr, nullptr);
     947             :         int ir_present =
     948          43 :             nc_inq_att(ncid, varid, CF_SG_INTERIOR_RING, nullptr, nullptr);
     949             : 
     950          43 :         if (pnc_present == NC_ENOTATT && ir_present == NC_ENOTATT)
     951             :         {
     952          10 :             ret = POLYGON;
     953             :         }
     954             :         else
     955          33 :             ret = MULTIPOLYGON;
     956             :     }
     957             : 
     958          72 :     return ret;
     959             : }
     960             : 
     961         133 : void inPlaceSerialize_Point(SGeometry_Reader *ge, size_t seek_pos,
     962             :                             std::vector<unsigned char> &buffer)
     963             : {
     964         133 :     uint8_t order = PLATFORM_HEADER;
     965         187 :     uint32_t t = ge->get_axisCount() == 2   ? wkbPoint
     966          54 :                  : ge->get_axisCount() == 3 ? wkbPoint25D
     967         133 :                                             : wkbNone;
     968             : 
     969         133 :     if (t == wkbNone)
     970           0 :         throw SG_Exception_BadFeature();
     971             : 
     972         133 :     add_to_buffer(buffer, order);
     973         133 :     add_to_buffer(buffer, t);
     974             : 
     975             :     // Now get point data;
     976         133 :     Point &p = (*ge)[seek_pos];
     977         133 :     add_to_buffer(buffer, p[0]);
     978         133 :     add_to_buffer(buffer, p[1]);
     979             : 
     980         133 :     if (ge->get_axisCount() >= 3)
     981             :     {
     982          54 :         add_to_buffer(buffer, p[2]);
     983             :     }
     984         133 : }
     985             : 
     986          45 : void inPlaceSerialize_LineString(SGeometry_Reader *ge, int node_count,
     987             :                                  size_t seek_begin,
     988             :                                  std::vector<unsigned char> &buffer)
     989             : {
     990          45 :     uint8_t order = PLATFORM_HEADER;
     991          65 :     uint32_t t = ge->get_axisCount() == 2   ? wkbLineString
     992          20 :                  : ge->get_axisCount() == 3 ? wkbLineString25D
     993          45 :                                             : wkbNone;
     994             : 
     995          45 :     if (t == wkbNone)
     996           0 :         throw SG_Exception_BadFeature();
     997          45 :     uint32_t nc = (uint32_t)node_count;
     998             : 
     999          45 :     add_to_buffer(buffer, order);
    1000          45 :     add_to_buffer(buffer, t);
    1001          45 :     add_to_buffer(buffer, nc);
    1002             : 
    1003             :     // Now serialize points
    1004         170 :     for (int ind = 0; ind < node_count; ind++)
    1005             :     {
    1006         125 :         Point &p = (*ge)[seek_begin + ind];
    1007         125 :         add_to_buffer(buffer, p[0]);
    1008         125 :         add_to_buffer(buffer, p[1]);
    1009             : 
    1010         125 :         if (ge->get_axisCount() >= 3)
    1011             :         {
    1012          56 :             add_to_buffer(buffer, p[2]);
    1013             :         }
    1014             :     }
    1015          45 : }
    1016             : 
    1017         432 : void inPlaceSerialize_PolygonExtOnly(SGeometry_Reader *ge, int node_count,
    1018             :                                      size_t seek_begin,
    1019             :                                      std::vector<unsigned char> &buffer)
    1020             : {
    1021         432 :     uint8_t order = PLATFORM_HEADER;
    1022         444 :     uint32_t t = ge->get_axisCount() == 2   ? wkbPolygon
    1023          12 :                  : ge->get_axisCount() == 3 ? wkbPolygon25D
    1024         432 :                                             : wkbNone;
    1025             : 
    1026         432 :     if (t == wkbNone)
    1027           0 :         throw SG_Exception_BadFeature();
    1028         432 :     uint32_t rc = 1;
    1029             : 
    1030         432 :     add_to_buffer(buffer, order);
    1031         432 :     add_to_buffer(buffer, t);
    1032         432 :     add_to_buffer(buffer, rc);
    1033         432 :     add_to_buffer(buffer, static_cast<uint32_t>(node_count));
    1034       47383 :     for (int pind = 0; pind < node_count; pind++)
    1035             :     {
    1036       46951 :         Point &pt = (*ge)[seek_begin + pind];
    1037       46951 :         add_to_buffer(buffer, pt[0]);
    1038       46951 :         add_to_buffer(buffer, pt[1]);
    1039             : 
    1040       46951 :         if (ge->get_axisCount() >= 3)
    1041             :         {
    1042          51 :             add_to_buffer(buffer, pt[2]);
    1043             :         }
    1044             :     }
    1045         432 : }
    1046             : 
    1047         166 : void inPlaceSerialize_Polygon(SGeometry_Reader *ge, std::vector<int> &pnc,
    1048             :                               int ring_count, size_t seek_begin,
    1049             :                               std::vector<unsigned char> &buffer)
    1050             : {
    1051         166 :     uint8_t order = PLATFORM_HEADER;
    1052         180 :     uint32_t t = ge->get_axisCount() == 2   ? wkbPolygon
    1053          14 :                  : ge->get_axisCount() == 3 ? wkbPolygon25D
    1054         166 :                                             : wkbNone;
    1055             : 
    1056         166 :     if (t == wkbNone)
    1057           0 :         throw SG_Exception_BadFeature();
    1058         166 :     uint32_t rc = static_cast<uint32_t>(ring_count);
    1059             : 
    1060         166 :     add_to_buffer(buffer, order);
    1061         166 :     add_to_buffer(buffer, t);
    1062         166 :     add_to_buffer(buffer, rc);
    1063         166 :     int cmoffset = 0;
    1064         366 :     for (int ring_c = 0; ring_c < ring_count; ring_c++)
    1065             :     {
    1066         200 :         uint32_t node_count = pnc[ring_c];
    1067         200 :         add_to_buffer(buffer, node_count);
    1068             : 
    1069         200 :         int pind = 0;
    1070       20332 :         for (pind = 0; pind < pnc[ring_c]; pind++)
    1071             :         {
    1072       20132 :             Point &pt = (*ge)[seek_begin + cmoffset + pind];
    1073       20132 :             add_to_buffer(buffer, pt[0]);
    1074       20132 :             add_to_buffer(buffer, pt[1]);
    1075             : 
    1076       20132 :             if (ge->get_axisCount() >= 3)
    1077             :             {
    1078          78 :                 add_to_buffer(buffer, pt[2]);
    1079             :             }
    1080             :         }
    1081             : 
    1082         200 :         cmoffset += pind;
    1083             :     }
    1084         166 : }
    1085             : 
    1086          75 : int scanForGeometryContainers(int ncid, std::set<int> &r_ids)
    1087             : {
    1088             :     int nvars;
    1089          75 :     if (nc_inq_nvars(ncid, &nvars) != NC_NOERR)
    1090             :     {
    1091           0 :         return -1;
    1092             :     }
    1093             : 
    1094          75 :     r_ids.clear();
    1095             : 
    1096             :     // For each variable check for geometry attribute
    1097             :     // If has geometry attribute, then check the associated variable ID
    1098             : 
    1099         703 :     for (int itr = 0; itr < nvars; itr++)
    1100             :     {
    1101             :         char c[NC_MAX_CHAR];
    1102         628 :         memset(c, 0, NC_MAX_CHAR);
    1103         628 :         if (nc_get_att_text(ncid, itr, CF_SG_GEOMETRY, c) != NC_NOERR)
    1104             :         {
    1105         513 :             continue;
    1106             :         }
    1107             : 
    1108             :         int varID;
    1109         115 :         if (nc_inq_varid(ncid, c, &varID) != NC_NOERR)
    1110             :         {
    1111           0 :             continue;
    1112             :         }
    1113             : 
    1114             :         // Now have variable ID. See if vector contains it, and if not
    1115             :         // insert
    1116         115 :         r_ids.insert(varID);  // It's a set. No big deal sets don't allow
    1117             :                               // duplicates anyways
    1118             :     }
    1119             : 
    1120          75 :     return 0;
    1121             : }
    1122             : }  // namespace nccfdriver

Generated by: LCOV version 1.14