LCOV - code coverage report
Current view: top level - tools - DLLoader.cpp (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 22 38 57.9 %
Date: 2025-03-25 09:33:27 Functions: 5 8 62.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 "DLLoader.h"
      23             : 
      24             : #include "Exception.h"
      25             : #include <cstdlib>
      26             : #include <iostream>
      27             : #include "core/ActionRegister.h"
      28             : #include "core/CLToolRegister.h"
      29             : 
      30             : #ifdef __PLUMED_HAS_DLOPEN
      31             : #include <dlfcn.h>
      32             : #endif
      33             : 
      34             : namespace PLMD {
      35             : 
      36          42 : bool DLLoader::installed() {
      37             : #ifdef __PLUMED_HAS_DLOPEN
      38          42 :   return true;
      39             : #else
      40             :   return false;
      41             : #endif
      42             : }
      43             : 
      44             : 
      45          51 : void* DLLoader::load(const std::string&s) {
      46             : #ifdef __PLUMED_HAS_DLOPEN
      47          51 :   auto lockerAction=Register::registrationLock(s);
      48          51 :   void* p=dlopen(s.c_str(),RTLD_NOW|RTLD_LOCAL);
      49          51 :   if(!p) {
      50           0 :     plumed_error()<<"Could not load library "<<s<<"\n"<<dlerror();
      51             :   }
      52          51 :   handles.push_back(p);
      53          51 :   Register::completeAllRegistrations(p);
      54          51 :   return p;
      55             : #else
      56             :   plumed_error()<<"you are trying to use dlopen but it's not configured on your system";
      57             : #endif
      58          51 : }
      59             : 
      60      812284 : DLLoader::~DLLoader() {
      61      812284 :   auto debug=std::getenv("PLUMED_LOAD_DEBUG");
      62             : #ifdef __PLUMED_HAS_DLOPEN
      63      812284 :   if(debug) {
      64           0 :     std::fprintf(stderr,"delete dlloader\n");
      65             :   }
      66      812335 :   while(!handles.empty()) {
      67          51 :     int ret=dlclose(handles.back());
      68          51 :     if(ret) {
      69           0 :       std::fprintf(stderr,"+++ error reported by dlclose: %s\n",dlerror());
      70             :     }
      71             :     handles.pop_back();
      72             :   }
      73      812284 :   if(debug) {
      74           0 :     std::fprintf(stderr,"end delete dlloader\n");
      75             :   }
      76             : #endif
      77      812284 : }
      78             : 
      79      812284 : DLLoader::DLLoader() {
      80             :   // do nothing
      81      812284 : }
      82             : 
      83       55012 : const std::vector<void*> & DLLoader::getHandles() const noexcept {
      84       55012 :   return handles;
      85             : }
      86             : 
      87           0 : DLLoader::EnsureGlobalDLOpen::EnsureGlobalDLOpen(const void *symbol) noexcept {
      88             : #ifdef __PLUMED_HAS_DLOPEN
      89             : #ifdef __PLUMED_HAS_DLADDR
      90             :   Dl_info info;
      91             :   // from the manual:
      92             :   // If the address specified in addr could not be matched to a shared
      93             :   //      object, then these functions return 0.  In this case, an error
      94             :   //      message is not available via dlerror(3).
      95           0 :   int zeroIsError=dladdr(symbol, &info);
      96           0 :   if(zeroIsError!=0) {
      97             :     //This "promotes" to GLOBAL the object with the symbol pointed by ptr
      98           0 :     handle_ = dlopen(info.dli_fname, RTLD_GLOBAL | RTLD_NOW);
      99             :   } else {
     100           0 :     std::fprintf(stderr,
     101             :                  "+++WARNING+++"
     102             :                  "Failure in finding any object that contains the symbol %p.\n",
     103             :                  symbol);
     104             :   }
     105             : #else
     106             :   std::fprintf(stderr,
     107             :                "+++WARNING+++"
     108             :                "I can't use dladdr for promoting the library containing the symbol %p.\n"
     109             :                "This system seems not to support dladdr",
     110             :                symbol);
     111             : #endif //__PLUMED_HAS_DLADDR
     112             : #endif //__PLUMED_HAS_DLOPEN
     113           0 : }
     114             : 
     115           0 : DLLoader::EnsureGlobalDLOpen::~EnsureGlobalDLOpen() {
     116             : #ifdef __PLUMED_HAS_DLOPEN
     117           0 :   if (handle_) {
     118           0 :     dlclose(handle_);
     119             :   }
     120             : #endif //__PLUMED_HAS_DLOPEN
     121           0 : }
     122             : 
     123           0 : bool DLLoader::isPlumedGlobal() {
     124             : #if defined(__APPLE__)
     125             :   bool result;
     126             : #ifdef __PLUMED_HAS_DLOPEN
     127             :   void* handle;
     128             : #if defined(__PLUMED_HAS_RTLD_DEFAULT)
     129             :   handle=RTLD_DEFAULT;
     130             : #else
     131             :   handle=dlopen(NULL,RTLD_LOCAL);
     132             : #endif
     133             :   // we check for two variants, see wrapper/Plumed.h for an explanation:
     134             :   result=dlsym(handle,"plumed_plumedmain_create") || dlsym(handle,"plumedmain_create");
     135             :   if(handle) {
     136             :     dlclose(handle);
     137             :   }
     138             : #else
     139             :   // if a system cannot use dlopen, we assume plumed is globally available
     140             :   result=true;
     141             : #endif //__PLUMED_HAS_DLOPEN
     142             :   return result;
     143             : #else
     144           0 :   plumed_error()<<"DLLoader::isPlumedGlobal() is only functional with APPLE dlsym";
     145             : #endif
     146             : }
     147             : 
     148             : } // namespace PLMD

Generated by: LCOV version 1.16