LCOV - code coverage report
Current view: top level - core - PlumedMainInitializer.cpp (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 259 320 80.9 %
Date: 2025-03-25 09:33:27 Functions: 14 16 87.5 %

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2011-2023 The plumed team
       3             :    (see the PEOPLE file at the root of the distribution for a list of names)
       4             : 
       5             :    See http://www.plumed.org for more information.
       6             : 
       7             :    This file is part of plumed, version 2.
       8             : 
       9             :    plumed is free software: you can redistribute it and/or modify
      10             :    it under the terms of the GNU Lesser General Public License as published by
      11             :    the Free Software Foundation, either version 3 of the License, or
      12             :    (at your option) any later version.
      13             : 
      14             :    plumed is distributed in the hope that it will be useful,
      15             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      16             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      17             :    GNU Lesser General Public License for more details.
      18             : 
      19             :    You should have received a copy of the GNU Lesser General Public License
      20             :    along with plumed.  If not, see <http://www.gnu.org/licenses/>.
      21             : +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
      22             : #include "PlumedMainInitializer.h"
      23             : #include "PlumedMain.h"
      24             : #include "tools/Exception.h"
      25             : #include "lepton/Exception.h"
      26             : #include <cstdlib>
      27             : #include <cstring>
      28             : #include <iostream>
      29             : #if defined __PLUMED_HAS_DLOPEN
      30             : #include <dlfcn.h>
      31             : #endif
      32             : #include <exception>
      33             : #include <stdexcept>
      34             : #include <ios>
      35             : #include <new>
      36             : #include <typeinfo>
      37             : #include <system_error>
      38             : #include <future>
      39             : #include <memory>
      40             : #include <functional>
      41             : #include <regex>
      42             : #include <any>
      43             : #include <variant>
      44             : #include <optional>
      45             : #include <filesystem>
      46             : #include "tools/TypesafePtr.h"
      47             : #include "tools/Log.h"
      48             : #include "tools/Tools.h"
      49             : 
      50             : 
      51       20449 : static bool getenvTypesafeDebug() noexcept {
      52       20449 :   static const auto* res=std::getenv("PLUMED_TYPESAFE_DEBUG");
      53       20449 :   return res;
      54             : }
      55             : 
      56             : // cppcheck-suppress passedByValue
      57           0 : static void typesafeDebug(const char*key,plumed_safeptr_x safe) noexcept {
      58           0 :   std::fprintf(stderr,"+++ PLUMED_TYPESAFE_DEBUG %s %p %zu",key,safe.ptr,safe.nelem);
      59           0 :   const size_t* shape=safe.shape;
      60           0 :   if(shape) {
      61           0 :     std::fprintf(stderr," (");
      62           0 :     while(*shape!=0) {
      63           0 :       std::fprintf(stderr," %zu",*shape);
      64           0 :       shape++;
      65             :     }
      66           0 :     std::fprintf(stderr," )");
      67             :   }
      68           0 :   std::fprintf(stderr," %zx %p\n",safe.flags,safe.opt);
      69           0 : }
      70             : 
      71             : // create should never throw
      72             : // in case of a problem, it logs the error and return a null pointer
      73             : // when loaded by an interface >=2.5, this will result in a non valid plumed object.
      74             : // earlier interfaces will just give a segfault or a failed assertion.
      75      805720 : extern "C" void*plumed_plumedmain_create() {
      76             :   try {
      77      805720 :     return new PLMD::PlumedMain;
      78           0 :   } catch(const std::exception & e) {
      79           0 :     std::cerr<<"+++ an error happened while creating a plumed object\n";
      80           0 :     std::cerr<<e.what()<<std::endl;
      81             :     return nullptr;
      82           0 :   } catch(...) {
      83           0 :     std::cerr<<"+++ an unknown error happened while creating a plumed object"<<std::endl;
      84             :     return nullptr;
      85           0 :   }
      86             : }
      87             : 
      88     8000671 : extern "C" unsigned plumed_plumedmain_create_reference(void*plumed) {
      89     8000671 :   plumed_massert(plumed,"trying to create a reference to a plumed object which is not initialized");
      90             :   auto p=static_cast<PLMD::PlumedMain*>(plumed);
      91     8000671 :   return p->increaseReferenceCounter();
      92             : }
      93             : 
      94     8806391 : extern "C" unsigned plumed_plumedmain_delete_reference(void*plumed) {
      95     8806391 :   plumed_massert(plumed,"trying to delete a reference to a plumed object which is not initialized");
      96             :   auto p=static_cast<PLMD::PlumedMain*>(plumed);
      97     8806391 :   return p->decreaseReferenceCounter();
      98             : }
      99             : 
     100          42 : extern "C" unsigned plumed_plumedmain_use_count(void*plumed) {
     101          42 :   plumed_massert(plumed,"trying to delete a reference to a plumed object which is not initialized");
     102             :   auto p=static_cast<PLMD::PlumedMain*>(plumed);
     103          42 :   return p->useCountReferenceCounter();
     104             : }
     105             : 
     106         357 : extern "C" void plumed_plumedmain_cmd(void*plumed,const char*key,const void*val) {
     107         357 :   plumed_massert(plumed,"trying to use a plumed object which is not initialized");
     108             :   auto p=static_cast<PLMD::PlumedMain*>(plumed);
     109         357 :   p->cmd(key,PLMD::TypesafePtr::unchecked(val));
     110         357 : }
     111             : 
     112             : extern "C" {
     113          95 :   static void plumed_plumedmain_cmd_safe(void*plumed,const char*key,plumed_safeptr_x safe) {
     114          95 :     plumed_massert(plumed,"trying to use a plumed object which is not initialized");
     115             :     auto p=static_cast<PLMD::PlumedMain*>(plumed);
     116          95 :     if(getenvTypesafeDebug()) {
     117           0 :       typesafeDebug(key,safe);
     118             :     }
     119          95 :     p->cmd(key,PLMD::TypesafePtr::fromSafePtr(&safe));
     120          95 :   }
     121             : }
     122             : 
     123             : /// Internal tool
     124             : /// Throws the currently managed exception and call the nothrow handler.
     125             : /// If nested is not null, it is passed and then gets populated with a pointer that should
     126             : /// be called on the nested exception
     127             : /// If msg is not null, it overrides the message. Can be used to build a concatenated message.
     128         178 : static void translate_current(plumed_nothrow_handler_x nothrow,void**nested=nullptr,const char*msg=nullptr) {
     129         178 :   const void* opt[11]= {"n",nested,nullptr,nullptr,nullptr,nullptr,nullptr,nullptr,nullptr,nullptr,nullptr};
     130             :   try {
     131             :     // this function needs to be called while catching an exception
     132             :     // cppcheck-suppress rethrowNoCurrentException
     133         178 :     throw;
     134         178 :   } catch(const PLMD::ExceptionTypeError & e) {
     135          52 :     if(!msg) {
     136           0 :       msg=e.what();
     137             :     }
     138          52 :     nothrow.handler(nothrow.ptr,20300,msg,opt);
     139         105 :   } catch(const PLMD::ExceptionError & e) {
     140          53 :     if(!msg) {
     141           2 :       msg=e.what();
     142             :     }
     143          53 :     nothrow.handler(nothrow.ptr,20200,msg,opt);
     144          54 :   } catch(const PLMD::ExceptionDebug & e) {
     145           1 :     if(!msg) {
     146           1 :       msg=e.what();
     147             :     }
     148           1 :     nothrow.handler(nothrow.ptr,20100,msg,opt);
     149          24 :   } catch(const PLMD::Exception & e) {
     150          23 :     if(!msg) {
     151          16 :       msg=e.what();
     152             :     }
     153          23 :     nothrow.handler(nothrow.ptr,20000,msg,opt);
     154          24 :   } catch(const PLMD::lepton::Exception & e) {
     155           1 :     if(!msg) {
     156           1 :       msg=e.what();
     157             :     }
     158           1 :     nothrow.handler(nothrow.ptr,19900,msg,opt);
     159             :     // 11000 to 12000 are "bad exceptions". message will be copied without new allocations
     160           2 :   } catch(const std::bad_variant_access & e) {
     161           1 :     if(!msg) {
     162           1 :       msg=e.what();
     163             :     }
     164           1 :     nothrow.handler(nothrow.ptr,11700,msg,opt);
     165           2 :   } catch(const std::bad_optional_access & e) {
     166           1 :     if(!msg) {
     167           1 :       msg=e.what();
     168             :     }
     169           1 :     nothrow.handler(nothrow.ptr,11600,msg,opt);
     170           2 :   } catch(const std::bad_exception & e) {
     171           1 :     if(!msg) {
     172           1 :       msg=e.what();
     173             :     }
     174           1 :     nothrow.handler(nothrow.ptr,11500,msg,opt);
     175           2 :   } catch(const std::bad_array_new_length & e) {
     176           1 :     if(!msg) {
     177           1 :       msg=e.what();
     178             :     }
     179           1 :     nothrow.handler(nothrow.ptr,11410,msg,opt);
     180           4 :   } catch(const std::bad_alloc & e) {
     181           3 :     if(!msg) {
     182           3 :       msg=e.what();
     183             :     }
     184           3 :     nothrow.handler(nothrow.ptr,11400,msg,opt);
     185           4 :   } catch(const std::bad_function_call & e) {
     186           1 :     if(!msg) {
     187           1 :       msg=e.what();
     188             :     }
     189           1 :     nothrow.handler(nothrow.ptr,11300,msg,opt);
     190           2 :   } catch(const std::bad_weak_ptr & e) {
     191           1 :     if(!msg) {
     192           1 :       msg=e.what();
     193             :     }
     194           1 :     nothrow.handler(nothrow.ptr,11200,msg,opt);
     195           2 :   } catch(const std::bad_any_cast & e) {
     196           1 :     if(!msg) {
     197           1 :       msg=e.what();
     198             :     }
     199           1 :     nothrow.handler(nothrow.ptr,11150,msg,opt);
     200           2 :   } catch(const std::bad_cast & e) {
     201           1 :     if(!msg) {
     202           1 :       msg=e.what();
     203             :     }
     204           1 :     nothrow.handler(nothrow.ptr,11100,msg,opt);
     205           2 :   } catch(const std::bad_typeid & e) {
     206           1 :     if(!msg) {
     207           1 :       msg=e.what();
     208             :     }
     209           1 :     nothrow.handler(nothrow.ptr,11000,msg,opt);
     210          14 :   } catch(const std::regex_error & e) {
     211          13 :     if(!msg) {
     212          13 :       msg=e.what();
     213             :     }
     214          13 :     if(e.code()==std::regex_constants::error_collate) {
     215           1 :       nothrow.handler(nothrow.ptr,10240,msg,opt);
     216             :     } else if(e.code()==std::regex_constants::error_ctype) {
     217           1 :       nothrow.handler(nothrow.ptr,10241,msg,opt);
     218             :     } else if(e.code()==std::regex_constants::error_escape) {
     219           1 :       nothrow.handler(nothrow.ptr,10242,msg,opt);
     220             :     } else if(e.code()==std::regex_constants::error_backref) {
     221           1 :       nothrow.handler(nothrow.ptr,10243,msg,opt);
     222             :     } else if(e.code()==std::regex_constants::error_brack) {
     223           1 :       nothrow.handler(nothrow.ptr,10244,msg,opt);
     224             :     } else if(e.code()==std::regex_constants::error_paren) {
     225           1 :       nothrow.handler(nothrow.ptr,10245,msg,opt);
     226             :     } else if(e.code()==std::regex_constants::error_brace) {
     227           1 :       nothrow.handler(nothrow.ptr,10246,msg,opt);
     228             :     } else if(e.code()==std::regex_constants::error_badbrace) {
     229           1 :       nothrow.handler(nothrow.ptr,10247,msg,opt);
     230             :     } else if(e.code()==std::regex_constants::error_range) {
     231           1 :       nothrow.handler(nothrow.ptr,10248,msg,opt);
     232             :     } else if(e.code()==std::regex_constants::error_space) {
     233           1 :       nothrow.handler(nothrow.ptr,10249,msg,opt);
     234             :     } else if(e.code()==std::regex_constants::error_badrepeat) {
     235           1 :       nothrow.handler(nothrow.ptr,10250,msg,opt);
     236             :     } else if(e.code()==std::regex_constants::error_complexity) {
     237           1 :       nothrow.handler(nothrow.ptr,10251,msg,opt);
     238             :     } else if(e.code()==std::regex_constants::error_stack) {
     239           1 :       nothrow.handler(nothrow.ptr,10252,msg,opt);
     240             :     }
     241             :     // fallback to generic runtime_error
     242             :     else {
     243           0 :       nothrow.handler(nothrow.ptr,10200,msg,opt);
     244             :     }
     245          16 :   } catch(const std::filesystem::filesystem_error & e) {
     246           3 :     if(!msg) {
     247           3 :       msg=e.what();
     248             :     }
     249           3 :     int value=e.code().value();
     250             : 
     251           3 :     opt[2]="c"; // "c" passes the error code.
     252           3 :     opt[3]=&value;
     253             : 
     254           3 :     opt[4]="C"; // "C" passes the error category
     255           3 :     int generic_category=1;
     256           3 :     int system_category=2;
     257           3 :     int iostream_category=3;
     258           3 :     int future_category=4;
     259           3 :     if(e.code().category()==std::generic_category()) {
     260           3 :       opt[5]=&generic_category;
     261           0 :     } else if(e.code().category()==std::system_category()) {
     262           0 :       opt[5]=&system_category;
     263           0 :     } else if(e.code().category()==std::iostream_category()) {
     264           0 :       opt[5]=&iostream_category;
     265           0 :     } else if(e.code().category()==std::future_category()) {
     266           0 :       opt[5]=&future_category;
     267             :     } else {
     268           0 :       opt[5]=nullptr;
     269             :     }
     270             : 
     271             :     // local class, just needed to propely pass path information
     272             :     // path is stored as a span of bytes.
     273             :     // in theory, could be wchar type (on Windows, not tested),
     274             :     // so we explicitly store the number of bytes
     275             :     struct Path {
     276             :       std::size_t numbytes=0;
     277             :       const void* ptr=nullptr;
     278             :       Path(const std::filesystem::path & path):
     279             :         numbytes(sizeof(std::filesystem::path::value_type)*path.native().length()),
     280             :         ptr(path.c_str())
     281             :       {}
     282             :       Path() = default;
     283             :     };
     284             : 
     285           3 :     opt[6]="p"; // path1
     286           3 :     Path path1; // declared here since it should survive till the end of this function
     287           3 :     if(!e.path1().empty()) {
     288           2 :       path1=Path(e.path1());
     289           2 :       opt[7]=&path1;
     290             :     }
     291             : 
     292           3 :     opt[8]="q"; // path2
     293           3 :     Path path2; // declared here since it should survive till the end of this function
     294           3 :     if(!e.path2().empty()) {
     295           1 :       path2=Path(e.path2());
     296           1 :       opt[9]=&path2;
     297             :     }
     298             : 
     299           3 :     nothrow.handler(nothrow.ptr,10229,msg,opt);
     300           4 :   } catch(const std::ios_base::failure & e) {
     301           1 :     if(!msg) {
     302           1 :       msg=e.what();
     303             :     }
     304           1 :     int value=e.code().value();
     305           1 :     opt[2]="c"; // "c" passes the error code.
     306           1 :     opt[3]=&value;
     307           1 :     if(e.code().category()==std::generic_category()) {
     308           0 :       nothrow.handler(nothrow.ptr,10230,msg,opt);
     309           1 :     } else if(e.code().category()==std::system_category()) {
     310           0 :       nothrow.handler(nothrow.ptr,10231,msg,opt);
     311           1 :     } else if(e.code().category()==std::iostream_category()) {
     312           1 :       nothrow.handler(nothrow.ptr,10232,msg,opt);
     313           0 :     } else if(e.code().category()==std::future_category()) {
     314           0 :       nothrow.handler(nothrow.ptr,10233,msg,opt);
     315             :     } else
     316             :       // 10239 represents std::ios_base::failure with default constructur
     317             :     {
     318           0 :       nothrow.handler(nothrow.ptr,10239,msg,opt);
     319             :     }
     320           5 :   } catch(const std::system_error & e) {
     321           4 :     if(!msg) {
     322           4 :       msg=e.what();
     323             :     }
     324           4 :     int value=e.code().value();
     325           4 :     opt[2]="c"; // "c" passes the error code.
     326           4 :     opt[3]=&value;
     327           4 :     if(e.code().category()==std::generic_category()) {
     328           1 :       nothrow.handler(nothrow.ptr,10220,msg,opt);
     329           3 :     } else if(e.code().category()==std::system_category()) {
     330           1 :       nothrow.handler(nothrow.ptr,10221,msg,opt);
     331           2 :     } else if(e.code().category()==std::iostream_category()) {
     332           1 :       nothrow.handler(nothrow.ptr,10222,msg,opt);
     333           1 :     } else if(e.code().category()==std::future_category()) {
     334           1 :       nothrow.handler(nothrow.ptr,10223,msg,opt);
     335             :     }
     336             :     // fallback to generic runtime_error
     337             :     else {
     338           0 :       nothrow.handler(nothrow.ptr,10200,msg,opt);
     339             :     }
     340           5 :   } catch(const std::underflow_error &e) {
     341           1 :     if(!msg) {
     342           1 :       msg=e.what();
     343             :     }
     344           1 :     nothrow.handler(nothrow.ptr,10215,msg,opt);
     345           2 :   } catch(const std::overflow_error &e) {
     346           1 :     if(!msg) {
     347           1 :       msg=e.what();
     348             :     }
     349           1 :     nothrow.handler(nothrow.ptr,10210,msg,opt);
     350           2 :   } catch(const std::range_error &e) {
     351           1 :     if(!msg) {
     352           1 :       msg=e.what();
     353             :     }
     354           1 :     nothrow.handler(nothrow.ptr,10205,msg,opt);
     355           2 :   } catch(const std::runtime_error & e) {
     356           1 :     if(!msg) {
     357           1 :       msg=e.what();
     358             :     }
     359           1 :     nothrow.handler(nothrow.ptr,10200,msg,opt);
     360           5 :   } catch(const std::future_error & e) {
     361           4 :     if(!msg) {
     362           4 :       msg=e.what();
     363             :     }
     364             :     if(e.code()==std::make_error_code(std::future_errc::broken_promise)) {
     365           1 :       nothrow.handler(nothrow.ptr,10125,msg,opt);
     366             :     } else if(e.code()==std::make_error_code(std::future_errc::future_already_retrieved)) {
     367           1 :       nothrow.handler(nothrow.ptr,10126,msg,opt);
     368             :     } else if(e.code()==std::make_error_code(std::future_errc::promise_already_satisfied)) {
     369           1 :       nothrow.handler(nothrow.ptr,10127,msg,opt);
     370             :     } else if(e.code()==std::make_error_code(std::future_errc::no_state)) {
     371           1 :       nothrow.handler(nothrow.ptr,10128,msg,opt);
     372             :     }
     373             :     // fallback to generic logic_error
     374             :     else {
     375           0 :       nothrow.handler(nothrow.ptr,10100,msg,opt);
     376             :     }
     377           5 :   } catch(const std::out_of_range & e) {
     378           1 :     if(!msg) {
     379           1 :       msg=e.what();
     380             :     }
     381           1 :     nothrow.handler(nothrow.ptr,10120,msg,opt);
     382           2 :   } catch(const std::length_error & e) {
     383           1 :     if(!msg) {
     384           1 :       msg=e.what();
     385             :     }
     386           1 :     nothrow.handler(nothrow.ptr,10115,msg,opt);
     387           2 :   } catch(const std::domain_error & e) {
     388           1 :     if(!msg) {
     389           1 :       msg=e.what();
     390             :     }
     391           1 :     nothrow.handler(nothrow.ptr,10110,msg,opt);
     392           2 :   } catch(const std::invalid_argument & e) {
     393           1 :     if(!msg) {
     394           1 :       msg=e.what();
     395             :     }
     396           1 :     nothrow.handler(nothrow.ptr,10105,msg,opt);
     397           2 :   } catch(const std::logic_error & e) {
     398           1 :     if(!msg) {
     399           1 :       msg=e.what();
     400             :     }
     401           1 :     nothrow.handler(nothrow.ptr,10100,msg,opt);
     402             :     // generic exception. message will be copied without new allocations
     403             :     // reports all non caught exceptions that are derived from std::exception
     404             :     // for instance, boost exceptions would end up here
     405           1 :   } catch(const std::exception & e) {
     406           0 :     if(!msg) {
     407           0 :       msg=e.what();
     408             :     }
     409           0 :     nothrow.handler(nothrow.ptr,10000,msg,opt);
     410           1 :   } catch(const char* m) {
     411           1 :     if(!msg) {
     412             :       msg=m;
     413             :     }
     414           1 :     nothrow.handler(nothrow.ptr,10000,msg,opt);
     415           1 :   } catch(const std::string & s) {
     416           0 :     if(!msg) {
     417             :       msg=s.c_str();
     418             :     }
     419           0 :     nothrow.handler(nothrow.ptr,10000,msg,opt);
     420           1 :   } catch (...) {
     421             :     // if exception cannot be translated, we add a bad_exception to the stack
     422           1 :     nothrow.handler(nothrow.ptr,11500,"plumed could not translate exception",opt);
     423           1 :   }
     424         178 : }
     425             : 
     426          68 : static void translate_nested(plumed_nothrow_handler_x nothrow) {
     427             :   try {
     428          68 :     throw;
     429          68 :   } catch (const std::nested_exception & e) {
     430             : // If this exception has a nested one:
     431          12 :     auto nothrow_nested=nothrow;
     432          12 :     nothrow_nested.ptr=nullptr;
     433             : // translate the current exception asking the wrapper to allocate a new exception
     434          12 :     translate_current(nothrow,&nothrow_nested.ptr);
     435             : // if the wrapper cannot allocate the exception, this will be a nullptr
     436          12 :     if(nothrow_nested.ptr) {
     437             :       try {
     438             : // transfer control to the nested exception
     439          12 :         e.rethrow_nested();
     440          12 :       } catch (...) {
     441             : // recursively translate it
     442          12 :         translate_nested(nothrow_nested);
     443          12 :       }
     444             :     }
     445          68 :   } catch (...) {
     446             : // otherwise, just translate the current exception
     447          56 :     translate_current(nothrow);
     448          56 :   }
     449          68 : }
     450             : 
     451             : extern "C" {
     452       20354 :   static void plumed_plumedmain_cmd_safe_nothrow(void*plumed,const char*key,plumed_safeptr_x safe,plumed_nothrow_handler_x nothrow) {
     453             : // This is a workaround for a suboptimal choice in PLUMED <2.8
     454             : // In particular, the only way to bypass the exception handling process was to call the plumed_plumedmain_cmd_safe
     455             : // function directly.
     456             : // With this modification, it is possible to just call the plumed_plumedmain_cmd_safe_nothrow function
     457             : // passing a null error handler.
     458       20354 :     if(!nothrow.handler) {
     459           0 :       plumed_plumedmain_cmd_safe(plumed,key,safe);
     460           0 :       return;
     461             :     }
     462             :     auto p=static_cast<PLMD::PlumedMain*>(plumed);
     463             : // At library boundaries we translate exceptions to error codes.
     464             : // This allows an exception to be catched also if the MD code
     465             : // was linked against a different C++ library
     466             :     try {
     467       20354 :       plumed_massert(plumed,"trying to use a plumed object which is not initialized");
     468       20354 :       if(getenvTypesafeDebug()) {
     469           0 :         typesafeDebug(key,safe);
     470             :       }
     471       20354 :       p->cmd(key,PLMD::TypesafePtr::fromSafePtr(&safe));
     472         166 :     } catch(...) {
     473         166 :       if(p->getNestedExceptions()) {
     474          56 :         translate_nested(nothrow);
     475             :       } else {
     476             : // In this case, we just consider the latest thrown exception and
     477             : // supplement it with a concatenated message so as not to
     478             : // loose information.
     479         110 :         auto msg=PLMD::Tools::concatenateExceptionMessages();
     480         110 :         translate_current(nothrow,nullptr,msg.c_str());
     481             :       }
     482         166 :     }
     483             :   }
     484             : }
     485             : 
     486             : extern "C" {
     487           0 :   static void plumed_plumedmain_cmd_nothrow(void*plumed,const char*key,const void*val,plumed_nothrow_handler_x nothrow) {
     488             :     plumed_safeptr_x safe;
     489           0 :     plumed_assert(nothrow.handler) << "Accepting a null pointer here would make the calling code non compatible with plumed 2.5 to 2.7";
     490           0 :     safe.ptr=val;
     491           0 :     safe.nelem=0;
     492           0 :     safe.shape=NULL;
     493           0 :     safe.flags=0;
     494           0 :     safe.opt=NULL;
     495           0 :     plumed_plumedmain_cmd_safe_nothrow(plumed,key,safe,nothrow);
     496           0 :   }
     497             : }
     498             : 
     499      805720 : extern "C" void plumed_plumedmain_finalize(void*plumed) {
     500      805720 :   plumed_massert(plumed,"trying to deallocate a plumed object which is not initialized");
     501             : // I think it is not possible to replace this delete with a smart pointer
     502             : // since the ownership of this pointer is in a C structure. GB
     503      805720 :   delete static_cast<PLMD::PlumedMain*>(plumed);
     504      805720 : }
     505             : 
     506             : // values here should be consistent with those in plumed_symbol_table_init !!!!
     507             : plumed_symbol_table_type_x plumed_symbol_table= {
     508             :   4,
     509             :   {plumed_plumedmain_create,plumed_plumedmain_cmd,plumed_plumedmain_finalize},
     510             :   plumed_plumedmain_cmd_nothrow,
     511             :   plumed_plumedmain_cmd_safe,
     512             :   plumed_plumedmain_cmd_safe_nothrow,
     513             :   plumed_plumedmain_create_reference,
     514             :   plumed_plumedmain_delete_reference,
     515             :   plumed_plumedmain_use_count
     516             : };
     517             : 
     518             : // values here should be consistent with those above !!!!
     519      811107 : extern "C" void plumed_symbol_table_init() {
     520      811107 :   plumed_symbol_table.version=4;
     521      811107 :   plumed_symbol_table.functions.create=plumed_plumedmain_create;
     522      811107 :   plumed_symbol_table.functions.cmd=plumed_plumedmain_cmd;
     523      811107 :   plumed_symbol_table.functions.finalize=plumed_plumedmain_finalize;
     524      811107 :   plumed_symbol_table.cmd_nothrow=plumed_plumedmain_cmd_nothrow;
     525      811107 :   plumed_symbol_table.cmd_safe=plumed_plumedmain_cmd_safe;
     526      811107 :   plumed_symbol_table.cmd_safe_nothrow=plumed_plumedmain_cmd_safe_nothrow;
     527      811107 :   plumed_symbol_table.create_reference=plumed_plumedmain_create_reference;
     528      811107 :   plumed_symbol_table.delete_reference=plumed_plumedmain_delete_reference;
     529      811107 :   plumed_symbol_table.use_count=plumed_plumedmain_use_count;
     530      811107 : }
     531             : 
     532             : namespace PLMD {
     533             : 
     534             : #define plumed_convert_fptr(ptr,fptr) { ptr=NULL; std::memcpy(&ptr,&fptr,(sizeof(fptr)>sizeof(ptr)?sizeof(ptr):sizeof(fptr))); }
     535             : 
     536             : /// Static object which registers Plumed.
     537             : /// This is a static object which, during its construction at startup,
     538             : /// registers the pointers to plumed_plumedmain_create, plumed_plumedmain_cmd and plumed_plumedmain_finalize
     539             : /// to the plumed_kernel_register function.
     540             : /// Registration is only required with plumed loader <=2.4, but we do it anyway in order to maintain
     541             : /// backward compatibility. Notice that as of plumed 2.5 the plumed_kernel_register is found
     542             : /// using dlsym, in order to allow the libplumedKernel library to be loadable also when
     543             : /// the plumed_kernel_register symbol is not available.
     544             : namespace {
     545             : class PlumedMainInitializer {
     546             :   const bool debug;
     547             : public:
     548        5418 :   PlumedMainInitializer():
     549        5418 :     debug(std::getenv("PLUMED_LOAD_DEBUG")) {
     550             : // make sure static plumed_function_pointers is initialized here
     551        5418 :     plumed_symbol_table_init();
     552        5418 :     if(debug) {
     553           0 :       std::fprintf(stderr,"+++ Initializing PLUMED with plumed_symbol_table version %i at %p\n",plumed_symbol_table.version,(void*)&plumed_symbol_table);
     554             :     }
     555             : #if defined(__PLUMED_HAS_DLOPEN)
     556        5418 :     if(std::getenv("PLUMED_LOAD_SKIP_REGISTRATION")) {
     557           0 :       if(debug) {
     558           0 :         std::fprintf(stderr,"+++ Skipping registration +++\n");
     559             :       }
     560           0 :       return;
     561             :     }
     562             :     typedef plumed_plumedmain_function_holder_x* (*plumed_kernel_register_type_x)(const plumed_plumedmain_function_holder_x*);
     563             :     plumed_kernel_register_type_x plumed_kernel_register=nullptr;
     564             :     void* handle=nullptr;
     565             : #if defined(__PLUMED_HAS_RTLD_DEFAULT)
     566        5418 :     if(debug) {
     567           0 :       std::fprintf(stderr,"+++ Registering functions. Looking in RTLD_DEFAULT +++\n");
     568             :     }
     569        5418 :     void* dls=dlsym(RTLD_DEFAULT,"plumed_kernel_register");
     570             : #else
     571             :     handle=dlopen(NULL,RTLD_LOCAL);
     572             :     if(debug) {
     573             :       std::fprintf(stderr,"+++ Registering functions. dlopen handle at %p +++\n",handle);
     574             :     }
     575             :     void* dls=dlsym(handle,"plumed_kernel_register");
     576             : #endif
     577        5418 :     *(void **)(&plumed_kernel_register)=dls;
     578        5418 :     if(debug) {
     579           0 :       if(plumed_kernel_register) {
     580           0 :         std::fprintf(stderr,"+++ plumed_kernel_register found at %p +++\n",dls);
     581             :       } else {
     582           0 :         std::fprintf(stderr,"+++ plumed_kernel_register not found +++\n");
     583             :       }
     584             :     }
     585             :     void*createp;
     586             :     void*cmdp;
     587             :     void*finalizep;
     588             :     plumed_convert_fptr(createp,plumed_symbol_table.functions.create);
     589             :     plumed_convert_fptr(cmdp,plumed_symbol_table.functions.cmd);
     590             :     plumed_convert_fptr(finalizep,plumed_symbol_table.functions.finalize);
     591        5418 :     if(plumed_kernel_register && debug)
     592           0 :       std::fprintf(stderr,"+++ Registering functions at %p (%p,%p,%p) +++\n",
     593             :                    (void*)&plumed_symbol_table.functions,createp,cmdp,finalizep);
     594        5418 :     if(plumed_kernel_register) {
     595          26 :       (*plumed_kernel_register)(&plumed_symbol_table.functions);
     596             :     }
     597             : // Notice that handle could be null in the following cases:
     598             : // - if we use RTLD_DEFAULT
     599             : // - on Linux if we don't use RTLD_DEFAULT, since dlopen(NULL,RTLD_LOCAL) returns a null pointer.
     600             :     if(handle) {
     601             :       dlclose(handle);
     602             :     }
     603             : #endif
     604             :   }
     605        5418 :   ~PlumedMainInitializer() {
     606        5418 :     if(debug) {
     607           0 :       std::fprintf(stderr,"+++ Finalizing PLUMED with plumed_symbol_table at %p\n",(void*)&plumed_symbol_table);
     608             :     }
     609        5418 :   }
     610             : } PlumedMainInitializerRegisterMe;
     611             : }
     612             : 
     613             : }
     614             : 
     615             : 

Generated by: LCOV version 1.16