EVOLUTION-MANAGER
Edit File: ogr_proj_p.cpp
/****************************************************************************** * * Project: GDAL * Purpose: PROJ-related functionality * Author: Even Rouault <even dot rouault at spatialys dot com> * ****************************************************************************** * Copyright (c) 2018, Even Rouault <even dot rouault at spatialys dot com> * * 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. ****************************************************************************/ #include "cpl_error.h" #include "cpl_multiproc.h" #include "cpl_string.h" #include "ogr_proj_p.h" #include "ogr_srs_api.h" #include "proj.h" #include <mutex> /*! @cond Doxygen_Suppress */ static void osr_proj_logger(void * /* user_data */, int level, const char * message) { if( level == PJ_LOG_ERROR ) { CPLError(CE_Failure, CPLE_AppDefined, "PROJ: %s", message); } else if( level == PJ_LOG_DEBUG ) { CPLDebug("PROJ", "%s", message); } else if( level == PJ_LOG_TRACE ) { CPLDebug("PROJ_TRACE", "%s", message); } } static unsigned g_searchPathGenerationCounter = 0; static std::mutex g_oSearchPathMutex; static CPLStringList g_aosSearchpaths; struct OSRPJContextHolder { unsigned searchPathGenerationCounter = 0; PJ_CONTEXT* context = nullptr; OSRProjTLSCache oCache{}; OSRPJContextHolder() = default; ~OSRPJContextHolder(); void init(); void deinit(); private: OSRPJContextHolder(const OSRPJContextHolder&) = delete; OSRPJContextHolder& operator=(const OSRPJContextHolder&) = delete; }; void OSRPJContextHolder::init() { if( !context ) { context = proj_context_create(); proj_log_func (context, nullptr, osr_proj_logger); } } OSRPJContextHolder::~OSRPJContextHolder() { deinit(); } void OSRPJContextHolder::deinit() { oCache.clear(); // Destroy context in last proj_context_destroy(context); context = nullptr; } #ifdef WIN32 // Currently thread_local and C++ objects don't work well with DLL on Windows static void FreeProjTLSContextHolder( void* pData ) { delete static_cast<OSRPJContextHolder*>(pData); } static OSRPJContextHolder& GetProjTLSContextHolder() { static OSRPJContextHolder dummy; int bMemoryErrorOccurred = false; void* pData = CPLGetTLSEx(CTLS_PROJCONTEXTHOLDER, &bMemoryErrorOccurred); if( bMemoryErrorOccurred ) { return dummy; } if( pData == nullptr) { auto pHolder = new OSRPJContextHolder(); CPLSetTLSWithFreeFuncEx( CTLS_PROJCONTEXTHOLDER, pHolder, FreeProjTLSContextHolder, &bMemoryErrorOccurred ); if( bMemoryErrorOccurred ) { delete pHolder; return dummy; } return *pHolder; } return *static_cast<OSRPJContextHolder*>(pData); } #else static thread_local OSRPJContextHolder g_tls_projContext; static OSRPJContextHolder& GetProjTLSContextHolder() { return g_tls_projContext; } #endif PJ_CONTEXT* OSRGetProjTLSContext() { auto& l_projContext = GetProjTLSContextHolder(); l_projContext.init(); { // If OSRSetPROJSearchPaths() has been called since we created the context, // set the new search paths on the context. std::lock_guard<std::mutex> oLock(g_oSearchPathMutex); if( l_projContext.searchPathGenerationCounter != g_searchPathGenerationCounter ) { l_projContext.searchPathGenerationCounter = g_searchPathGenerationCounter; proj_context_set_search_paths( l_projContext.context, g_aosSearchpaths.Count(), g_aosSearchpaths.List()); } } return l_projContext.context; } /************************************************************************/ /* OSRGetProjTLSCache() */ /************************************************************************/ OSRProjTLSCache* OSRGetProjTLSCache() { auto& l_projContext = GetProjTLSContextHolder(); return &l_projContext.oCache; } struct OSRPJDeleter { void operator()(PJ* pj) const { proj_destroy(pj); } }; void OSRProjTLSCache::clear() { m_oCacheEPSG.clear(); m_oCacheWKT.clear(); } PJ* OSRProjTLSCache::GetPJForEPSGCode(int nCode, bool bUseNonDeprecated, bool bAddTOWGS84) { try { const EPSGCacheKey key(nCode, bUseNonDeprecated, bAddTOWGS84); const auto& cached = m_oCacheEPSG.get(key); return proj_clone(OSRGetProjTLSContext(), cached.get()); } catch( const lru11::KeyNotFound& ) { return nullptr; } } void OSRProjTLSCache::CachePJForEPSGCode(int nCode, bool bUseNonDeprecated, bool bAddTOWGS84, PJ* pj) { const EPSGCacheKey key(nCode, bUseNonDeprecated, bAddTOWGS84); m_oCacheEPSG.insert(key, std::shared_ptr<PJ>( proj_clone(OSRGetProjTLSContext(), pj), OSRPJDeleter())); } PJ* OSRProjTLSCache::GetPJForWKT(const std::string& wkt) { try { const auto& cached = m_oCacheWKT.get(wkt); return proj_clone(OSRGetProjTLSContext(), cached.get()); } catch( const lru11::KeyNotFound& ) { return nullptr; } } void OSRProjTLSCache::CachePJForWKT(const std::string& wkt, PJ* pj) { m_oCacheWKT.insert(wkt, std::shared_ptr<PJ>( proj_clone(OSRGetProjTLSContext(), pj), OSRPJDeleter())); } /************************************************************************/ /* OSRCleanupTLSContext() */ /************************************************************************/ void OSRCleanupTLSContext() { GetProjTLSContextHolder().deinit(); } /*! @endcond */ /************************************************************************/ /* OSRSetPROJSearchPaths() */ /************************************************************************/ /** \brief Set the search path(s) for PROJ resource files. * * @param papszPaths NULL terminated list of directory paths. * @since GDAL 3.0 */ void OSRSetPROJSearchPaths( const char* const * papszPaths ) { std::lock_guard<std::mutex> oLock(g_oSearchPathMutex); g_searchPathGenerationCounter ++; g_aosSearchpaths.Assign(CSLDuplicate(papszPaths), true); } /************************************************************************/ /* OSRGetPROJSearchPaths() */ /************************************************************************/ /** \brief Get the search path(s) for PROJ resource files. * * @return NULL terminated list of directory paths. To be freed with CSLDestroy() * @since GDAL 3.0.3 */ char** OSRGetPROJSearchPaths() { std::lock_guard<std::mutex> oLock(g_oSearchPathMutex); const char* pszSep = #ifdef _WIN32 ";" #else ":" #endif ; return CSLTokenizeString2( proj_info().searchpath, pszSep, 0); } /************************************************************************/ /* OSRGetPROJVersion() */ /************************************************************************/ /** \brief Get the PROJ version * * @param pnMajor Pointer to major version number, or NULL * @param pnMinor Pointer to minor version number, or NULL * @param pnPatch Pointer to patch version number, or NULL * @since GDAL 3.0.1 */ void OSRGetPROJVersion( int* pnMajor, int* pnMinor, int* pnPatch ) { auto info = proj_info(); if (pnMajor) *pnMajor = info.major; if (pnMinor) *pnMinor = info.minor; if (pnPatch) *pnPatch = info.patch; }