EVOLUTION-MANAGER
Edit File: 4D_api.cpp
/****************************************************************************** * Project: PROJ.4 * Purpose: Implement a (currently minimalistic) proj API based primarily * on the PJ_COORD 4D geodetic spatiotemporal data type. * * Author: Thomas Knudsen, thokn@sdfe.dk, 2016-06-09/2016-11-06 * ****************************************************************************** * Copyright (c) 2016, 2017 Thomas Knudsen/SDFE * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. *****************************************************************************/ #define FROM_PROJ_CPP #include <assert.h> #include <errno.h> #include <stddef.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #ifndef _MSC_VER #include <strings.h> #endif #include <algorithm> #include <limits> #include "proj.h" #include "proj_experimental.h" #include "proj_internal.h" #include "proj_math.h" #include "geodesic.h" #include "proj/common.hpp" #include "proj/coordinateoperation.hpp" #include "proj/internal/internal.hpp" #include "proj/internal/io_internal.hpp" using namespace NS_PROJ::internal; /* Initialize PJ_COORD struct */ PJ_COORD proj_coord (double x, double y, double z, double t) { PJ_COORD res; res.v[0] = x; res.v[1] = y; res.v[2] = z; res.v[3] = t; return res; } static PJ_DIRECTION opposite_direction(PJ_DIRECTION dir) { return static_cast<PJ_DIRECTION>(-dir); } /*****************************************************************************/ int proj_angular_input (PJ *P, enum PJ_DIRECTION dir) { /****************************************************************************** Returns 1 if the operator P expects angular input coordinates when operating in direction dir, 0 otherwise. dir: {PJ_FWD, PJ_INV} ******************************************************************************/ if (PJ_FWD==dir) return pj_left (P)==PJ_IO_UNITS_RADIANS; return pj_right (P)==PJ_IO_UNITS_RADIANS; } /*****************************************************************************/ int proj_angular_output (PJ *P, enum PJ_DIRECTION dir) { /****************************************************************************** Returns 1 if the operator P provides angular output coordinates when operating in direction dir, 0 otherwise. dir: {PJ_FWD, PJ_INV} ******************************************************************************/ return proj_angular_input (P, opposite_direction(dir)); } /* Geodesic distance (in meter) + fwd and rev azimuth between two points on the ellipsoid */ PJ_COORD proj_geod (const PJ *P, PJ_COORD a, PJ_COORD b) { PJ_COORD c; if( !P->geod ) { return proj_coord_error(); } /* Note: the geodesic code takes arguments in degrees */ geod_inverse (P->geod, PJ_TODEG(a.lpz.phi), PJ_TODEG(a.lpz.lam), PJ_TODEG(b.lpz.phi), PJ_TODEG(b.lpz.lam), c.v, c.v+1, c.v+2 ); return c; } /* Geodesic distance (in meter) between two points with angular 2D coordinates */ double proj_lp_dist (const PJ *P, PJ_COORD a, PJ_COORD b) { double s12, azi1, azi2; /* Note: the geodesic code takes arguments in degrees */ if( !P->geod ) { return HUGE_VAL; } geod_inverse (P->geod, PJ_TODEG(a.lpz.phi), PJ_TODEG(a.lpz.lam), PJ_TODEG(b.lpz.phi), PJ_TODEG(b.lpz.lam), &s12, &azi1, &azi2 ); return s12; } /* The geodesic distance AND the vertical offset */ double proj_lpz_dist (const PJ *P, PJ_COORD a, PJ_COORD b) { if (HUGE_VAL==a.lpz.lam || HUGE_VAL==b.lpz.lam) return HUGE_VAL; return hypot (proj_lp_dist (P, a, b), a.lpz.z - b.lpz.z); } /* Euclidean distance between two points with linear 2D coordinates */ double proj_xy_dist (PJ_COORD a, PJ_COORD b) { return hypot (a.xy.x - b.xy.x, a.xy.y - b.xy.y); } /* Euclidean distance between two points with linear 3D coordinates */ double proj_xyz_dist (PJ_COORD a, PJ_COORD b) { return hypot (proj_xy_dist (a, b), a.xyz.z - b.xyz.z); } /* Measure numerical deviation after n roundtrips fwd-inv (or inv-fwd) */ double proj_roundtrip (PJ *P, PJ_DIRECTION direction, int n, PJ_COORD *coord) { int i; PJ_COORD t, org; if (nullptr==P) return HUGE_VAL; if (n < 1) { proj_errno_set (P, EINVAL); return HUGE_VAL; } /* in the first half-step, we generate the output value */ org = *coord; *coord = proj_trans (P, direction, org); t = *coord; /* now we take n-1 full steps in inverse direction: We are */ /* out of phase due to the half step already taken */ for (i = 0; i < n - 1; i++) t = proj_trans (P, direction, proj_trans (P, opposite_direction(direction), t) ); /* finally, we take the last half-step */ t = proj_trans (P, opposite_direction(direction), t); /* checking for angular *input* since we do a roundtrip, and end where we begin */ if (proj_angular_input (P, direction)) return proj_lpz_dist (P, org, t); return proj_xyz_dist (org, t); } /**************************************************************************************/ PJ_COORD proj_trans (PJ *P, PJ_DIRECTION direction, PJ_COORD coord) { /*************************************************************************************** Apply the transformation P to the coordinate coord, preferring the 4D interfaces if available. See also pj_approx_2D_trans and pj_approx_3D_trans in pj_internal.c, which work similarly, but prefers the 2D resp. 3D interfaces if available. ***************************************************************************************/ if (nullptr==P || direction == PJ_IDENT) return coord; if (P->inverted) direction = opposite_direction(direction); if( !P->alternativeCoordinateOperations.empty() ) { // Do a first pass and select the first coordinate operation whose area // of use is compatible with the input coordinate int i = 0; for( const auto &alt: P->alternativeCoordinateOperations ) { if( direction == PJ_FWD ) { if( coord.xyzt.x >= alt.minxSrc && coord.xyzt.y >= alt.minySrc && coord.xyzt.x <= alt.maxxSrc && coord.xyzt.y <= alt.maxySrc ) { if( P->iCurCoordOp != i ) { std::string msg("Using coordinate operation "); msg += alt.name; pj_log(P->ctx, PJ_LOG_TRACE, msg.c_str()); P->iCurCoordOp = i; } return pj_fwd4d( coord, alt.pj ); } } else { if( coord.xyzt.x >= alt.minxDst && coord.xyzt.y >= alt.minyDst && coord.xyzt.x <= alt.maxxDst && coord.xyzt.y <= alt.maxyDst ) { if( P->iCurCoordOp != i ) { std::string msg("Using coordinate operation "); msg += alt.name; pj_log(P->ctx, PJ_LOG_TRACE, msg.c_str()); P->iCurCoordOp = i; } return pj_inv4d( coord, alt.pj ); } } i ++; } // In case we did not find an operation whose area of use is compatible // with the input coordinate, then goes through again the list, and // use the first operation that does not require grids. i = 0; NS_PROJ::io::DatabaseContextPtr dbContext; try { if( P->ctx->cpp_context ) { dbContext = P->ctx->cpp_context->getDatabaseContext().as_nullable(); } } catch( const std::exception& ) {} for( const auto &alt: P->alternativeCoordinateOperations ) { auto coordOperation = dynamic_cast< NS_PROJ::operation::CoordinateOperation*>(alt.pj->iso_obj.get()); if( coordOperation ) { if( coordOperation->gridsNeeded(dbContext).empty() ) { if( P->iCurCoordOp != i ) { std::string msg("Using coordinate operation "); msg += alt.name; pj_log(P->ctx, PJ_LOG_TRACE, msg.c_str()); P->iCurCoordOp = i; } if( direction == PJ_FWD ) { return pj_fwd4d( coord, alt.pj ); } else { return pj_inv4d( coord, alt.pj ); } } } i++; } proj_errno_set (P, EINVAL); return proj_coord_error (); } switch (direction) { case PJ_FWD: return pj_fwd4d (coord, P); case PJ_INV: return pj_inv4d (coord, P); default: break; } proj_errno_set (P, EINVAL); return proj_coord_error (); } /*****************************************************************************/ int proj_trans_array (PJ *P, PJ_DIRECTION direction, size_t n, PJ_COORD *coord) { /****************************************************************************** Batch transform an array of PJ_COORD. Returns 0 if all coordinates are transformed without error, otherwise returns error number. ******************************************************************************/ size_t i; for (i = 0; i < n; i++) { coord[i] = proj_trans (P, direction, coord[i]); if (proj_errno(P)) return proj_errno (P); } return 0; } /*************************************************************************************/ size_t proj_trans_generic ( PJ *P, PJ_DIRECTION direction, double *x, size_t sx, size_t nx, double *y, size_t sy, size_t ny, double *z, size_t sz, size_t nz, double *t, size_t st, size_t nt ) { /************************************************************************************** Transform a series of coordinates, where the individual coordinate dimension may be represented by an array that is either 1. fully populated 2. a null pointer and/or a length of zero, which will be treated as a fully populated array of zeroes 3. of length one, i.e. a constant, which will be treated as a fully populated array of that constant value The strides, sx, sy, sz, st, represent the step length, in bytes, between consecutive elements of the corresponding array. This makes it possible for proj_transform to handle transformation of a large class of application specific data structures, without necessarily understanding the data structure format, as in: typedef struct {double x, y; int quality_level; char surveyor_name[134];} XYQS; XYQS survey[345]; double height = 23.45; PJ *P = {...}; size_t stride = sizeof (XYQS); ... proj_transform ( P, PJ_INV, sizeof(XYQS), &(survey[0].x), stride, 345, (* We have 345 eastings *) &(survey[0].y), stride, 345, (* ...and 345 northings. *) &height, 1, (* The height is the constant 23.45 m *) 0, 0 (* and the time is the constant 0.00 s *) ); This is similar to the inner workings of the pj_transform function, but the stride functionality has been generalized to work for any size of basic unit, not just a fixed number of doubles. In most cases, the stride will be identical for x, y,z, and t, since they will typically be either individual arrays (stride = sizeof(double)), or strided views into an array of application specific data structures (stride = sizeof (...)). But in order to support cases where x, y, z, and t come from heterogeneous sources, individual strides, sx, sy, sz, st, are used. Caveat: Since proj_transform does its work *in place*, this means that even the supposedly constants (i.e. length 1 arrays) will return from the call in altered state. Hence, remember to reinitialize between repeated calls. Return value: Number of transformations completed. **************************************************************************************/ PJ_COORD coord = {{0,0,0,0}}; size_t i, nmin; double null_broadcast = 0; double invalid_time = HUGE_VAL; if (nullptr==P) return 0; if (P->inverted) direction = opposite_direction(direction); /* ignore lengths of null arrays */ if (nullptr==x) nx = 0; if (nullptr==y) ny = 0; if (nullptr==z) nz = 0; if (nullptr==t) nt = 0; /* and make the nullities point to some real world memory for broadcasting nulls */ if (0==nx) x = &null_broadcast; if (0==ny) y = &null_broadcast; if (0==nz) z = &null_broadcast; if (0==nt) t = &invalid_time; /* nothing to do? */ if (0==nx+ny+nz+nt) return 0; /* arrays of length 1 are constants, which we broadcast along the longer arrays */ /* so we need to find the length of the shortest non-unity array to figure out */ /* how many coordinate pairs we must transform */ nmin = (nx > 1)? nx: (ny > 1)? ny: (nz > 1)? nz: (nt > 1)? nt: 1; if ((nx > 1) && (nx < nmin)) nmin = nx; if ((ny > 1) && (ny < nmin)) nmin = ny; if ((nz > 1) && (nz < nmin)) nmin = nz; if ((nt > 1) && (nt < nmin)) nmin = nt; /* Check validity of direction flag */ switch (direction) { case PJ_FWD: case PJ_INV: break; case PJ_IDENT: return nmin; default: proj_errno_set (P, EINVAL); return 0; } /* Arrays of length==0 are broadcast as the constant 0 */ /* Arrays of length==1 are broadcast as their single value */ /* Arrays of length >1 are iterated over (for the first nmin values) */ /* The slightly convolved incremental indexing is used due */ /* to the stride, which may be any size supported by the platform */ for (i = 0; i < nmin; i++) { coord.xyzt.x = *x; coord.xyzt.y = *y; coord.xyzt.z = *z; coord.xyzt.t = *t; coord = proj_trans(P, direction, coord); /* in all full length cases, we overwrite the input with the output, */ /* and step on to the next element. */ /* The casts are somewhat funky, but they compile down to no-ops and */ /* they tell compilers and static analyzers that we know what we do */ if (nx > 1) { *x = coord.xyzt.x; x = (double *) ((void *) ( ((char *) x) + sx)); } if (ny > 1) { *y = coord.xyzt.y; y = (double *) ((void *) ( ((char *) y) + sy)); } if (nz > 1) { *z = coord.xyzt.z; z = (double *) ((void *) ( ((char *) z) + sz)); } if (nt > 1) { *t = coord.xyzt.t; t = (double *) ((void *) ( ((char *) t) + st)); } } /* Last time around, we update the length 1 cases with their transformed alter egos */ if (nx==1) *x = coord.xyzt.x; if (ny==1) *y = coord.xyzt.y; if (nz==1) *z = coord.xyzt.z; if (nt==1) *t = coord.xyzt.t; return i; } /*************************************************************************************/ PJ_COORD pj_geocentric_latitude (const PJ *P, PJ_DIRECTION direction, PJ_COORD coord) { /************************************************************************************** Convert geographical latitude to geocentric (or the other way round if direction = PJ_INV) The conversion involves a call to the tangent function, which goes through the roof at the poles, so very close (the last centimeter) to the poles no conversion takes place and the input latitude is copied directly to the output. Fortunately, the geocentric latitude converges to the geographical at the poles, so the difference is negligible. For the spherical case, the geographical latitude equals the geocentric, and consequently, the input is copied directly to the output. **************************************************************************************/ const double limit = M_HALFPI - 1e-9; PJ_COORD res = coord; if ((coord.lp.phi > limit) || (coord.lp.phi < -limit) || (P->es==0)) return res; if (direction==PJ_FWD) res.lp.phi = atan (P->one_es * tan (coord.lp.phi) ); else res.lp.phi = atan (P->rone_es * tan (coord.lp.phi) ); return res; } double proj_torad (double angle_in_degrees) { return PJ_TORAD (angle_in_degrees);} double proj_todeg (double angle_in_radians) { return PJ_TODEG (angle_in_radians);} double proj_dmstor(const char *is, char **rs) { return dmstor(is, rs); } char* proj_rtodms(char *s, double r, int pos, int neg) { return rtodms(s, r, pos, neg); } /*************************************************************************************/ static PJ* skip_prep_fin(PJ *P) { /************************************************************************************** Skip prepare and finalize function for the various "helper operations" added to P when in cs2cs compatibility mode. **************************************************************************************/ P->skip_fwd_prepare = 1; P->skip_fwd_finalize = 1; P->skip_inv_prepare = 1; P->skip_inv_finalize = 1; return P; } /*************************************************************************************/ static int cs2cs_emulation_setup (PJ *P) { /************************************************************************************** If any cs2cs style modifiers are given (axis=..., towgs84=..., ) create the 4D API equivalent operations, so the preparation and finalization steps in the pj_inv/pj_fwd invocators can emulate the behaviour of pj_transform and the cs2cs app. Returns 1 on success, 0 on failure **************************************************************************************/ PJ *Q; paralist *p; int do_cart = 0; if (nullptr==P) return 0; /* Don't recurse when calling proj_create (which calls us back) */ if (pj_param_exists (P->params, "break_cs2cs_recursion")) return 1; /* Swap axes? */ p = pj_param_exists (P->params, "axis"); const bool disable_grid_presence_check = pj_param_exists ( P->params, "disable_grid_presence_check") != nullptr; /* Don't axisswap if data are already in "enu" order */ if (p && (0!=strcmp ("enu", p->param))) { char *def = static_cast<char*>(malloc (100+strlen(P->axis))); if (nullptr==def) return 0; sprintf (def, "break_cs2cs_recursion proj=axisswap axis=%s", P->axis); Q = pj_create_internal (P->ctx, def); free (def); if (nullptr==Q) return 0; P->axisswap = skip_prep_fin(Q); } /* Geoid grid(s) given? */ p = pj_param_exists (P->params, "geoidgrids"); if (!disable_grid_presence_check && p && strlen (p->param) > strlen ("geoidgrids=")) { char *gridnames = p->param + strlen ("geoidgrids="); char *def = static_cast<char*>(malloc (100+2*strlen(gridnames))); if (nullptr==def) return 0; sprintf (def, "break_cs2cs_recursion proj=vgridshift grids=%s", pj_double_quote_string_param_if_needed(gridnames).c_str()); Q = pj_create_internal (P->ctx, def); free (def); if (nullptr==Q) return 0; P->vgridshift = skip_prep_fin(Q); } /* Datum shift grid(s) given? */ p = pj_param_exists (P->params, "nadgrids"); if (!disable_grid_presence_check && p && strlen (p->param) > strlen ("nadgrids=")) { char *gridnames = p->param + strlen ("nadgrids="); char *def = static_cast<char*>(malloc (100+2*strlen(gridnames))); if (nullptr==def) return 0; sprintf (def, "break_cs2cs_recursion proj=hgridshift grids=%s", pj_double_quote_string_param_if_needed(gridnames).c_str()); Q = pj_create_internal (P->ctx, def); free (def); if (nullptr==Q) return 0; P->hgridshift = skip_prep_fin(Q); } /* We ignore helmert if we have grid shift */ p = P->hgridshift ? nullptr : pj_param_exists (P->params, "towgs84"); while (p) { char *def; char *s = p->param; double *d = P->datum_params; size_t n = strlen (s); /* We ignore null helmert shifts (common in auto-translated resource files, e.g. epsg) */ if (0==d[0] && 0==d[1] && 0==d[2] && 0==d[3] && 0==d[4] && 0==d[5] && 0==d[6]) { /* If the current ellipsoid is not WGS84, then make sure the */ /* change in ellipsoid is still done. */ if (!(fabs(P->a_orig - 6378137.0) < 1e-8 && fabs(P->es_orig - 0.0066943799901413) < 1e-15)) { do_cart = 1; } break; } if (n <= 8) /* 8==strlen ("towgs84=") */ return 0; def = static_cast<char*>(malloc (100+n)); if (nullptr==def) return 0; sprintf (def, "break_cs2cs_recursion proj=helmert exact %s convention=position_vector", s); Q = pj_create_internal (P->ctx, def); free(def); if (nullptr==Q) return 0; pj_inherit_ellipsoid_def (P, Q); P->helmert = skip_prep_fin (Q); break; } /* We also need cartesian/geographical transformations if we are working in */ /* geocentric/cartesian space or we need to do a Helmert transform. */ if (P->is_geocent || P->helmert || do_cart) { char def[150]; sprintf (def, "break_cs2cs_recursion proj=cart a=%40.20g es=%40.20g", P->a_orig, P->es_orig); { /* In case the current locale does not use dot but comma as decimal */ /* separator, replace it with dot, so that proj_atof() behaves */ /* correctly. */ /* TODO later: use C++ ostringstream with imbue(std::locale::classic()) */ /* to be locale unaware */ char* next_pos; for (next_pos = def; (next_pos = strchr (next_pos, ',')) != nullptr; next_pos++) { *next_pos = '.'; } } Q = pj_create_internal (P->ctx, def); if (nullptr==Q) return 0; P->cart = skip_prep_fin (Q); if (!P->is_geocent) { sprintf (def, "break_cs2cs_recursion proj=cart ellps=WGS84"); Q = pj_create_internal (P->ctx, def); if (nullptr==Q) return 0; P->cart_wgs84 = skip_prep_fin (Q); } } return 1; } /*************************************************************************************/ PJ *pj_create_internal (PJ_CONTEXT *ctx, const char *definition) { /*************************************************************************************/ /************************************************************************************** Create a new PJ object in the context ctx, using the given definition. If ctx==0, the default context is used, if definition==0, or invalid, a null-pointer is returned. The definition may use '+' as argument start indicator, as in "+proj=utm +zone=32", or leave it out, as in "proj=utm zone=32". It may even use free formatting "proj = utm; zone =32 ellps= GRS80". Note that the semicolon separator is allowed, but not required. **************************************************************************************/ PJ *P; char *args, **argv; size_t argc, n; int ret; int allow_init_epsg; if (nullptr==ctx) ctx = pj_get_default_ctx (); /* Make a copy that we can manipulate */ n = strlen (definition); args = (char *) malloc (n + 1); if (nullptr==args) { proj_context_errno_set(ctx, ENOMEM); return nullptr; } strcpy (args, definition); argc = pj_trim_argc (args); if (argc==0) { pj_dealloc (args); proj_context_errno_set(ctx, PJD_ERR_NO_ARGS); return nullptr; } argv = pj_trim_argv (argc, args); if (!argv) { pj_dealloc(args); proj_context_errno_set(ctx, ENOMEM); return nullptr; } /* ...and let pj_init_ctx do the hard work */ /* New interface: forbid init=epsg:XXXX syntax by default */ allow_init_epsg = proj_context_get_use_proj4_init_rules(ctx, FALSE); P = pj_init_ctx_with_allow_init_epsg (ctx, (int) argc, argv, allow_init_epsg); pj_dealloc (argv); pj_dealloc (args); /* Support cs2cs-style modifiers */ ret = cs2cs_emulation_setup (P); if (0==ret) return proj_destroy (P); return P; } /*************************************************************************************/ PJ *proj_create_argv (PJ_CONTEXT *ctx, int argc, char **argv) { /************************************************************************************** Create a new PJ object in the context ctx, using the given definition argument array argv. If ctx==0, the default context is used, if definition==0, or invalid, a null-pointer is returned. The definition arguments may use '+' as argument start indicator, as in {"+proj=utm", "+zone=32"}, or leave it out, as in {"proj=utm", "zone=32"}. **************************************************************************************/ PJ *P; const char *c; if (nullptr==ctx) ctx = pj_get_default_ctx (); if (nullptr==argv) { proj_context_errno_set(ctx, PJD_ERR_NO_ARGS); return nullptr; } /* We assume that free format is used, and build a full proj_create compatible string */ c = pj_make_args (argc, argv); if (nullptr==c) { proj_context_errno_set(ctx, ENOMEM); return nullptr; } P = proj_create (ctx, c); pj_dealloc ((char *) c); return P; } /*************************************************************************************/ PJ *pj_create_argv_internal (PJ_CONTEXT *ctx, int argc, char **argv) { /************************************************************************************** Same as proj_create_argv() but calls pj_create_internal() instead of proj_create() internally **************************************************************************************/ PJ *P; const char *c; if (nullptr==ctx) ctx = pj_get_default_ctx (); if (nullptr==argv) { proj_context_errno_set(ctx, PJD_ERR_NO_ARGS); return nullptr; } /* We assume that free format is used, and build a full proj_create compatible string */ c = pj_make_args (argc, argv); if (nullptr==c) { proj_context_errno_set(ctx, ENOMEM); return nullptr; } P = pj_create_internal (ctx, c); pj_dealloc ((char *) c); return P; } /** Create an area of use */ PJ_AREA * proj_area_create(void) { return static_cast<PJ_AREA*>(pj_calloc(1, sizeof(PJ_AREA))); } /** Assign a bounding box to an area of use. */ void proj_area_set_bbox(PJ_AREA *area, double west_lon_degree, double south_lat_degree, double east_lon_degree, double north_lat_degree) { area->bbox_set = TRUE; area->west_lon_degree = west_lon_degree; area->south_lat_degree = south_lat_degree; area->east_lon_degree = east_lon_degree; area->north_lat_degree = north_lat_degree; } /** Free an area of use */ void proj_area_destroy(PJ_AREA* area) { pj_dealloc(area); } /************************************************************************/ /* proj_context_use_proj4_init_rules() */ /************************************************************************/ void proj_context_use_proj4_init_rules(PJ_CONTEXT *ctx, int enable) { if( ctx == nullptr ) { ctx = pj_get_default_ctx(); } ctx->use_proj4_init_rules = enable; } /************************************************************************/ /* EQUAL() */ /************************************************************************/ static int EQUAL(const char* a, const char* b) { #ifdef _MSC_VER return _stricmp(a, b) == 0; #else return strcasecmp(a, b) == 0; #endif } /************************************************************************/ /* proj_context_get_use_proj4_init_rules() */ /************************************************************************/ int proj_context_get_use_proj4_init_rules(PJ_CONTEXT *ctx, int from_legacy_code_path) { const char* val = getenv("PROJ_USE_PROJ4_INIT_RULES"); if( ctx == nullptr ) { ctx = pj_get_default_ctx(); } if( val ) { if( EQUAL(val, "yes") || EQUAL(val, "on") || EQUAL(val, "true") ) { return TRUE; } if( EQUAL(val, "no") || EQUAL(val, "off") || EQUAL(val, "false") ) { return FALSE; } pj_log(ctx, PJ_LOG_ERROR, "Invalid value for PROJ_USE_PROJ4_INIT_RULES"); } if( ctx->use_proj4_init_rules >= 0 ) { return ctx->use_proj4_init_rules; } return from_legacy_code_path; } /** Adds a " +type=crs" suffix to a PROJ string (if it is a PROJ string) */ std::string pj_add_type_crs_if_needed(const std::string& str) { std::string ret(str); if( (starts_with(str, "proj=") || starts_with(str, "+proj=") || starts_with(str, "+init=") || starts_with(str, "+title=")) && str.find("type=crs") == std::string::npos ) { ret += " +type=crs"; } return ret; } /*****************************************************************************/ static void reproject_bbox(PJ* pjGeogToCrs, double west_lon, double south_lat, double east_lon, double north_lat, double& minx, double& miny, double& maxx, double& maxy) { /*****************************************************************************/ minx = -std::numeric_limits<double>::max(); miny = -std::numeric_limits<double>::max(); maxx = std::numeric_limits<double>::max(); maxy = std::numeric_limits<double>::max(); if( !(west_lon == -180.0 && east_lon == 180.0 && south_lat == -90.0 && north_lat == 90.0) ) { minx = -minx; miny = -miny; maxx = -maxx; maxy = -maxy; double x[21 * 4], y[21 * 4]; for( int j = 0; j <= 20; j++ ) { x[j] = west_lon + j * (east_lon - west_lon) / 20; y[j] = south_lat; x[21+j] = west_lon + j * (east_lon - west_lon) / 20; y[21+j] = north_lat; x[21*2+j] = west_lon; y[21*2+j] = south_lat + j * (north_lat - south_lat) / 20; x[21*3+j] = east_lon; y[21*3+j] = south_lat + j * (north_lat - south_lat) / 20; } proj_trans_generic ( pjGeogToCrs, PJ_FWD, x, sizeof(double), 21 * 4, y, sizeof(double), 21 * 4, nullptr, 0, 0, nullptr, 0, 0); for( int j = 0; j < 21 * 4; j++ ) { if( x[j] != HUGE_VAL && y[j] != HUGE_VAL ) { minx = std::min(minx, x[j]); miny = std::min(miny, y[j]); maxx = std::max(maxx, x[j]); maxy = std::max(maxy, y[j]); } } } } /*****************************************************************************/ static PJ* add_coord_op_to_list(PJ* op, double west_lon, double south_lat, double east_lon, double north_lat, PJ* pjGeogToSrc, PJ* pjGeogToDst, std::vector<PJconsts::CoordOperation>& altCoordOps) { /*****************************************************************************/ double minxSrc; double minySrc; double maxxSrc; double maxySrc; double minxDst; double minyDst; double maxxDst; double maxyDst; reproject_bbox(pjGeogToSrc, west_lon, south_lat, east_lon, north_lat, minxSrc, minySrc, maxxSrc, maxySrc); reproject_bbox(pjGeogToDst, west_lon, south_lat, east_lon, north_lat, minxDst, minyDst, maxxDst, maxyDst); if( minxSrc <= maxxSrc && minxDst <= maxxDst ) { const char* c_name = proj_get_name(op); std::string name(c_name ? c_name : ""); altCoordOps.emplace_back(minxSrc, minySrc, maxxSrc, maxySrc, minxDst, minyDst, maxxDst, maxyDst, op, name); op = nullptr; } return op; } /*****************************************************************************/ static PJ* create_operation_to_base_geog_crs(PJ_CONTEXT* ctx, const PJ* crs) { /*****************************************************************************/ // Create a geographic 2D long-lat degrees CRS that is related to the // CRS auto geodetic_crs = proj_crs_get_geodetic_crs(ctx, crs); if( !geodetic_crs ) { proj_context_log_debug(ctx, "Cannot find geodetic CRS matching CRS"); return nullptr; } auto geodetic_crs_type = proj_get_type(geodetic_crs); if( geodetic_crs_type == PJ_TYPE_GEOCENTRIC_CRS || geodetic_crs_type == PJ_TYPE_GEOGRAPHIC_2D_CRS || geodetic_crs_type == PJ_TYPE_GEOGRAPHIC_3D_CRS ) { auto datum = proj_crs_get_datum(ctx, geodetic_crs); if( datum ) { auto cs = proj_create_ellipsoidal_2D_cs( ctx, PJ_ELLPS2D_LONGITUDE_LATITUDE, nullptr, 0); auto temp = proj_create_geographic_crs_from_datum( ctx,"unnamed", datum, cs); proj_destroy(datum); proj_destroy(cs); proj_destroy(geodetic_crs); geodetic_crs = temp; geodetic_crs_type = proj_get_type(geodetic_crs); } } if( geodetic_crs_type != PJ_TYPE_GEOGRAPHIC_2D_CRS ) { // Shouldn't happen proj_context_log_debug(ctx, "Cannot find geographic CRS matching CRS"); proj_destroy(geodetic_crs); return nullptr; } // Create the transformation from this geographic 2D CRS to the source CRS auto operation_ctx = proj_create_operation_factory_context(ctx, nullptr); proj_operation_factory_context_set_spatial_criterion( ctx, operation_ctx, PROJ_SPATIAL_CRITERION_PARTIAL_INTERSECTION); proj_operation_factory_context_set_grid_availability_use( ctx, operation_ctx, PROJ_GRID_AVAILABILITY_DISCARD_OPERATION_IF_MISSING_GRID); auto op_list_to_geodetic = proj_create_operations( ctx, geodetic_crs, crs, operation_ctx); proj_operation_factory_context_destroy(operation_ctx); proj_destroy(geodetic_crs); if( op_list_to_geodetic == nullptr || proj_list_get_count(op_list_to_geodetic) == 0 ) { proj_context_log_debug(ctx, "Cannot compute transformation from geographic CRS to CRS"); proj_list_destroy(op_list_to_geodetic); return nullptr; } auto opGeogToCrs = proj_list_get(ctx, op_list_to_geodetic, 0); assert(opGeogToCrs); proj_list_destroy(op_list_to_geodetic); return opGeogToCrs; } /*****************************************************************************/ PJ *proj_create_crs_to_crs (PJ_CONTEXT *ctx, const char *source_crs, const char *target_crs, PJ_AREA *area) { /****************************************************************************** Create a transformation pipeline between two known coordinate reference systems. See docs/source/development/reference/functions.rst ******************************************************************************/ if( !ctx ) { ctx = pj_get_default_ctx(); } PJ* src; PJ* dst; try { std::string source_crs_modified(pj_add_type_crs_if_needed(source_crs)); std::string target_crs_modified(pj_add_type_crs_if_needed(target_crs)); src = proj_create(ctx, source_crs_modified.c_str()); if( !src ) { proj_context_log_debug(ctx, "Cannot instantiate source_crs"); return nullptr; } dst = proj_create(ctx, target_crs_modified.c_str()); if( !dst ) { proj_context_log_debug(ctx, "Cannot instantiate target_crs"); proj_destroy(src); return nullptr; } } catch( const std::exception& ) { return nullptr; } auto ret = proj_create_crs_to_crs_from_pj(ctx, src, dst, area, nullptr); proj_destroy(src); proj_destroy(dst); return ret; } /*****************************************************************************/ PJ *proj_create_crs_to_crs_from_pj (PJ_CONTEXT *ctx, const PJ *source_crs, const PJ *target_crs, PJ_AREA *area, const char* const *) { /****************************************************************************** Create a transformation pipeline between two known coordinate reference systems. See docs/source/development/reference/functions.rst ******************************************************************************/ if( !ctx ) { ctx = pj_get_default_ctx(); } auto operation_ctx = proj_create_operation_factory_context(ctx, nullptr); if( !operation_ctx ) { return nullptr; } if( area && area->bbox_set ) { proj_operation_factory_context_set_area_of_interest( ctx, operation_ctx, area->west_lon_degree, area->south_lat_degree, area->east_lon_degree, area->north_lat_degree); } proj_operation_factory_context_set_spatial_criterion( ctx, operation_ctx, PROJ_SPATIAL_CRITERION_PARTIAL_INTERSECTION); proj_operation_factory_context_set_grid_availability_use( ctx, operation_ctx, PROJ_GRID_AVAILABILITY_DISCARD_OPERATION_IF_MISSING_GRID); auto op_list = proj_create_operations(ctx, source_crs, target_crs, operation_ctx); if( !op_list ) { proj_operation_factory_context_destroy(operation_ctx); return nullptr; } auto op_count = proj_list_get_count(op_list); if( op_count == 0 ) { proj_list_destroy(op_list); proj_operation_factory_context_destroy(operation_ctx); proj_context_log_debug(ctx, "No operation found matching criteria"); return nullptr; } PJ* P = proj_list_get(ctx, op_list, 0); assert(P); if( P == nullptr || op_count == 1 || (area && area->bbox_set) || proj_get_type(source_crs) == PJ_TYPE_GEOCENTRIC_CRS || proj_get_type(target_crs) == PJ_TYPE_GEOCENTRIC_CRS ) { proj_list_destroy(op_list); proj_operation_factory_context_destroy(operation_ctx); return P; } auto pjGeogToSrc = create_operation_to_base_geog_crs(ctx, source_crs); if( !pjGeogToSrc ) { proj_list_destroy(op_list); proj_operation_factory_context_destroy(operation_ctx); proj_context_log_debug(ctx, "Cannot create transformation from geographic CRS of source CRS to source CRS"); proj_destroy(P); return nullptr; } auto pjGeogToDst = create_operation_to_base_geog_crs(ctx, target_crs); if( !pjGeogToDst ) { proj_list_destroy(op_list); proj_operation_factory_context_destroy(operation_ctx); proj_context_log_debug(ctx, "Cannot create transformation from geographic CRS of target CRS to target CRS"); proj_destroy(P); proj_destroy(pjGeogToSrc); return nullptr; } try { bool skipDefaultTransforms = true; // Iterate over source->target candidate transformations and reproject // their long-lat bounding box into the source CRS. for( int i = 0; i < op_count; i++ ) { auto op = proj_list_get(ctx, op_list, i); assert(op); double west_lon = 0.0; double south_lat = 0.0; double east_lon = 0.0; double north_lat = 0.0; const char* name = proj_get_name(op); bool canUseOp = true; if( skipDefaultTransforms && name && (strstr(name, "Ballpark geographic offset") || strstr(name, "Ballpark geocentric translation")) ) { // Skip default transformations unless there is already one at // the beginning (in which case all of them will have one) if( i == 0 ) { skipDefaultTransforms = false; } else { canUseOp = false; } } if( canUseOp && proj_get_area_of_use(ctx, op, &west_lon, &south_lat, &east_lon, &north_lat, nullptr) ) { if( west_lon <= east_lon ) { op = add_coord_op_to_list(op, west_lon, south_lat, east_lon, north_lat, pjGeogToSrc, pjGeogToDst, P->alternativeCoordinateOperations); } else { auto op_clone = proj_clone(ctx, op); op = add_coord_op_to_list(op, west_lon, south_lat, 180, north_lat, pjGeogToSrc, pjGeogToDst, P->alternativeCoordinateOperations); op_clone = add_coord_op_to_list(op_clone, -180, south_lat, east_lon, north_lat, pjGeogToSrc, pjGeogToDst, P->alternativeCoordinateOperations); proj_destroy(op_clone); } } proj_destroy(op); } proj_list_destroy(op_list); proj_operation_factory_context_destroy(operation_ctx); proj_destroy(pjGeogToSrc); proj_destroy(pjGeogToDst); // If there's finally juste a single result, return it directly if( P->alternativeCoordinateOperations.size() == 1 ) { auto retP = P->alternativeCoordinateOperations[0].pj; P->alternativeCoordinateOperations[0].pj = nullptr; proj_destroy(P); P = retP; } else { // The returned P is rather dummy P->iso_obj = nullptr; P->fwd = nullptr; P->inv = nullptr; P->fwd3d = nullptr; P->inv3d = nullptr; P->fwd4d = nullptr; P->inv4d = nullptr; } return P; } catch( const std::exception& ) { proj_list_destroy(op_list); proj_operation_factory_context_destroy(operation_ctx); proj_destroy(pjGeogToSrc); proj_destroy(pjGeogToDst); proj_destroy(P); return nullptr; } } PJ *proj_destroy (PJ *P) { pj_free (P); return nullptr; } /*****************************************************************************/ int proj_errno (const PJ *P) { /****************************************************************************** Read an error level from the context of a PJ. ******************************************************************************/ return pj_ctx_get_errno (pj_get_ctx ((PJ *) P)); } /*****************************************************************************/ int proj_context_errno (PJ_CONTEXT *ctx) { /****************************************************************************** Read an error directly from a context, without going through a PJ belonging to that context. ******************************************************************************/ if (nullptr==ctx) ctx = pj_get_default_ctx(); return pj_ctx_get_errno (ctx); } /*****************************************************************************/ int proj_errno_set (const PJ *P, int err) { /****************************************************************************** Set context-errno, bubble it up to the thread local errno, return err ******************************************************************************/ /* Use proj_errno_reset to explicitly clear the error status */ if (0==err) return 0; /* For P==0 err goes to the default context */ proj_context_errno_set (pj_get_ctx ((PJ *) P), err); errno = err; return err; } /*****************************************************************************/ int proj_errno_restore (const PJ *P, int err) { /****************************************************************************** Use proj_errno_restore when the current function succeeds, but the error flag was set on entry, and stored/reset using proj_errno_reset in order to monitor for new errors. See usage example under proj_errno_reset () ******************************************************************************/ if (0==err) return 0; proj_errno_set (P, err); return 0; } /*****************************************************************************/ int proj_errno_reset (const PJ *P) { /****************************************************************************** Clears errno in the context and thread local levels through the low level pj_ctx interface. Returns the previous value of the errno, for convenient reset/restore operations: int foo (PJ *P) { // errno may be set on entry, but we need to reset it to be able to // check for errors from "do_something_with_P(P)" int last_errno = proj_errno_reset (P); // local failure if (0==P) return proj_errno_set (P, 42); // call to function that may fail do_something_with_P (P); // failure in do_something_with_P? - keep latest error status if (proj_errno(P)) return proj_errno (P); // success - restore previous error status, return 0 return proj_errno_restore (P, last_errno); } ******************************************************************************/ int last_errno; last_errno = proj_errno (P); pj_ctx_set_errno (pj_get_ctx ((PJ *) P), 0); errno = 0; pj_errno = 0; return last_errno; } /* Create a new context */ PJ_CONTEXT *proj_context_create (void) { return pj_ctx_alloc (); } PJ_CONTEXT *proj_context_destroy (PJ_CONTEXT *ctx) { if (nullptr==ctx) return nullptr; /* Trying to free the default context is a no-op (since it is statically allocated) */ if (pj_get_default_ctx ()==ctx) return nullptr; pj_ctx_free (ctx); return nullptr; } /*****************************************************************************/ static char *path_append (char *buf, const char *app, size_t *buf_size) { /****************************************************************************** Helper for proj_info() below. Append app to buf, separated by a semicolon. Also handle allocation of longer buffer if needed. Returns buffer and adjusts *buf_size through provided pointer arg. ******************************************************************************/ char *p; size_t len, applen = 0, buflen = 0; #ifdef _WIN32 const char *delim = ";"; #else const char *delim = ":"; #endif /* Nothing to do? */ if (nullptr == app) return buf; applen = strlen (app); if (0 == applen) return buf; /* Start checking whether buf is long enough */ if (nullptr != buf) buflen = strlen (buf); len = buflen+applen+strlen (delim) + 1; /* "pj_realloc", so to speak */ if (*buf_size < len) { p = static_cast<char*>(pj_calloc (2 * len, sizeof (char))); if (nullptr==p) { pj_dealloc (buf); return nullptr; } *buf_size = 2 * len; if (buf != nullptr) strcpy (p, buf); pj_dealloc (buf); buf = p; } assert(buf); /* Only append a delimiter if something's already there */ if (0 != buflen) strcat (buf, delim); strcat (buf, app); return buf; } static const char *empty = {""}; static char version[64] = {""}; static PJ_INFO info = {0, 0, 0, nullptr, nullptr, nullptr, nullptr, 0}; /*****************************************************************************/ PJ_INFO proj_info (void) { /****************************************************************************** Basic info about the current instance of the PROJ.4 library. Returns PJ_INFO struct. ******************************************************************************/ size_t buf_size = 0; char *buf = nullptr; pj_acquire_lock (); info.major = PROJ_VERSION_MAJOR; info.minor = PROJ_VERSION_MINOR; info.patch = PROJ_VERSION_PATCH; /* This is a controlled environment, so no risk of sprintf buffer overflow. A normal version string is xx.yy.zz which is 8 characters long and there is room for 64 bytes in the version string. */ sprintf (version, "%d.%d.%d", info.major, info.minor, info.patch); info.version = version; info.release = pj_get_release (); /* build search path string */ auto ctx = pj_get_default_ctx(); if (!ctx || ctx->search_paths.empty()) { const char *envPROJ_LIB = getenv("PROJ_LIB"); buf = path_append(buf, envPROJ_LIB, &buf_size); #ifdef PROJ_LIB if (envPROJ_LIB == nullptr) { buf = path_append(buf, PROJ_LIB, &buf_size); } #endif } else { for (const auto &path : ctx->search_paths) { buf = path_append(buf, path.c_str(), &buf_size); } } pj_dalloc(const_cast<char*>(info.searchpath)); info.searchpath = buf ? buf : empty; info.paths = ctx ? ctx->c_compat_paths : nullptr; info.path_count = ctx ? static_cast<int>(ctx->search_paths.size()) : 0; pj_release_lock (); return info; } /*****************************************************************************/ PJ_PROJ_INFO proj_pj_info(PJ *P) { /****************************************************************************** Basic info about a particular instance of a projection object. Returns PJ_PROJ_INFO struct. ******************************************************************************/ PJ_PROJ_INFO pjinfo; char *def; memset(&pjinfo, 0, sizeof(PJ_PROJ_INFO)); pjinfo.accuracy = -1.0; if (nullptr==P) return pjinfo; /* coordinate operation description */ if( P->iCurCoordOp >= 0 ) { P = P->alternativeCoordinateOperations[P->iCurCoordOp].pj; } else if( !P->alternativeCoordinateOperations.empty() ) { pjinfo.id = "unknown"; pjinfo.description = "unavailable until proj_trans is called"; pjinfo.definition = "unavailable until proj_trans is called"; return pjinfo; } /* projection id */ if (pj_param(P->ctx, P->params, "tproj").i) pjinfo.id = pj_param(P->ctx, P->params, "sproj").s; if( P->iso_obj ) { pjinfo.description = P->iso_obj->nameStr().c_str(); } else { pjinfo.description = P->descr; } // accuracy if( P->iso_obj ) { auto conv = dynamic_cast<const NS_PROJ::operation::Conversion*>(P->iso_obj.get()); if( conv ) { pjinfo.accuracy = 0.0; } else { auto op = dynamic_cast<const NS_PROJ::operation::CoordinateOperation*>(P->iso_obj.get()); if( op ) { const auto& accuracies = op->coordinateOperationAccuracies(); if( !accuracies.empty() ) { try { pjinfo.accuracy = std::stod(accuracies[0]->value()); } catch ( const std::exception& ) {} } } } } /* projection definition */ if (P->def_full) def = P->def_full; else def = pj_get_def(P, 0); /* pj_get_def takes a non-const PJ pointer */ if (nullptr==def) pjinfo.definition = empty; else pjinfo.definition = pj_shrink (def); /* Make pj_free clean this up eventually */ P->def_full = def; pjinfo.has_inverse = pj_has_inverse(P); return pjinfo; } /*****************************************************************************/ PJ_GRID_INFO proj_grid_info(const char *gridname) { /****************************************************************************** Information about a named datum grid. Returns PJ_GRID_INFO struct. ******************************************************************************/ PJ_GRID_INFO grinfo; /*PJ_CONTEXT *ctx = proj_context_create(); */ PJ_CONTEXT *ctx = pj_get_default_ctx(); PJ_GRIDINFO *gridinfo = pj_gridinfo_init(ctx, gridname); memset(&grinfo, 0, sizeof(PJ_GRID_INFO)); /* in case the grid wasn't found */ if (gridinfo->filename == nullptr) { pj_gridinfo_free(ctx, gridinfo); strcpy(grinfo.format, "missing"); return grinfo; } /* The string copies below are automatically null-terminated due to */ /* the memset above, so strncpy is safe */ /* name of grid */ strncpy (grinfo.gridname, gridname, sizeof(grinfo.gridname) - 1); /* full path of grid */ pj_find_file(ctx, gridname, grinfo.filename, sizeof(grinfo.filename) - 1); /* grid format */ strncpy (grinfo.format, gridinfo->format, sizeof(grinfo.format) - 1); /* grid size */ grinfo.n_lon = gridinfo->ct->lim.lam; grinfo.n_lat = gridinfo->ct->lim.phi; /* cell size */ grinfo.cs_lon = gridinfo->ct->del.lam; grinfo.cs_lat = gridinfo->ct->del.phi; /* bounds of grid */ grinfo.lowerleft = gridinfo->ct->ll; grinfo.upperright.lam = grinfo.lowerleft.lam + grinfo.n_lon*grinfo.cs_lon; grinfo.upperright.phi = grinfo.lowerleft.phi + grinfo.n_lat*grinfo.cs_lat; pj_gridinfo_free(ctx, gridinfo); return grinfo; } /*****************************************************************************/ PJ_INIT_INFO proj_init_info(const char *initname){ /****************************************************************************** Information about a named init file. Maximum length of initname is 64. Returns PJ_INIT_INFO struct. If the init file is not found all members of the return struct are set to the empty string. If the init file is found, but the metadata is missing, the value is set to "Unknown". ******************************************************************************/ int file_found; char param[80], key[74]; paralist *start, *next; PJ_INIT_INFO ininfo; PJ_CONTEXT *ctx = pj_get_default_ctx(); memset(&ininfo, 0, sizeof(PJ_INIT_INFO)); file_found = pj_find_file(ctx, initname, ininfo.filename, sizeof(ininfo.filename)); if (!file_found || strlen(initname) > 64) { if( strcmp(initname, "epsg") == 0 || strcmp(initname, "EPSG") == 0 ) { const char* val; pj_ctx_set_errno( ctx, 0 ); strncpy (ininfo.name, initname, sizeof(ininfo.name) - 1); strcpy(ininfo.origin, "EPSG"); val = proj_context_get_database_metadata(ctx, "EPSG.VERSION"); if( val ) { strncpy(ininfo.version, val, sizeof(ininfo.version) - 1); } val = proj_context_get_database_metadata(ctx, "EPSG.DATE"); if( val ) { strncpy(ininfo.lastupdate, val, sizeof(ininfo.lastupdate) - 1); } return ininfo; } if( strcmp(initname, "IGNF") == 0 ) { const char* val; pj_ctx_set_errno( ctx, 0 ); strncpy (ininfo.name, initname, sizeof(ininfo.name) - 1); strcpy(ininfo.origin, "IGNF"); val = proj_context_get_database_metadata(ctx, "IGNF.VERSION"); if( val ) { strncpy(ininfo.version, val, sizeof(ininfo.version) - 1); } val = proj_context_get_database_metadata(ctx, "IGNF.DATE"); if( val ) { strncpy(ininfo.lastupdate, val, sizeof(ininfo.lastupdate) - 1); } return ininfo; } return ininfo; } /* The initial memset (0) makes strncpy safe here */ strncpy (ininfo.name, initname, sizeof(ininfo.name) - 1); strcpy(ininfo.origin, "Unknown"); strcpy(ininfo.version, "Unknown"); strcpy(ininfo.lastupdate, "Unknown"); strncpy (key, initname, 64); /* make room for ":metadata\0" at the end */ key[64] = 0; memcpy(key + strlen(key), ":metadata", 9 + 1); strcpy(param, "+init="); /* The +strlen(param) avoids a cppcheck false positive warning */ strncat(param + strlen(param), key, sizeof(param)-1-strlen(param)); start = pj_mkparam(param); pj_expand_init(ctx, start); if (pj_param(ctx, start, "tversion").i) strncpy(ininfo.version, pj_param(ctx, start, "sversion").s, sizeof(ininfo.version) - 1); if (pj_param(ctx, start, "torigin").i) strncpy(ininfo.origin, pj_param(ctx, start, "sorigin").s, sizeof(ininfo.origin) - 1); if (pj_param(ctx, start, "tlastupdate").i) strncpy(ininfo.lastupdate, pj_param(ctx, start, "slastupdate").s, sizeof(ininfo.lastupdate) - 1); for ( ; start; start = next) { next = start->next; pj_dalloc(start); } return ininfo; } /*****************************************************************************/ PJ_FACTORS proj_factors(PJ *P, PJ_COORD lp) { /****************************************************************************** Cartographic characteristics at point lp. Characteristics include meridian, parallel and areal scales, angular distortion, meridian/parallel, meridian convergence and scale error. returns PJ_FACTORS. If unsuccessful, error number is set and the struct returned contains NULL data. ******************************************************************************/ PJ_FACTORS factors = {0,0,0, 0,0,0, 0,0, 0,0,0,0}; struct FACTORS f; if (nullptr==P) return factors; if (pj_factors(lp.lp, P, 0.0, &f)) return factors; factors.meridional_scale = f.h; factors.parallel_scale = f.k; factors.areal_scale = f.s; factors.angular_distortion = f.omega; factors.meridian_parallel_angle = f.thetap; factors.meridian_convergence = f.conv; factors.tissot_semimajor = f.a; factors.tissot_semiminor = f.b; /* Raw derivatives, for completeness's sake */ factors.dx_dlam = f.der.x_l; factors.dx_dphi = f.der.x_p; factors.dy_dlam = f.der.y_l; factors.dy_dphi = f.der.y_p; return factors; }