LCOV - code coverage report
Current view: top level - tools - TypesafePtr.h (source / functions) Hit Total Coverage
Test: plumed test coverage Lines: 129 143 90.2 %
Date: 2025-03-25 09:33:27 Functions: 44 60 73.3 %

          Line data    Source code
       1             : /* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
       2             :    Copyright (c) 2018-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             : #ifndef __PLUMED_tools_TypesafePtr_h
      23             : #define __PLUMED_tools_TypesafePtr_h
      24             : 
      25             : #include "Exception.h"
      26             : 
      27             : #include <memory>
      28             : #include <iosfwd>
      29             : #include <map>
      30             : #include <utility>
      31             : #include <mutex>
      32             : #include <cstdio>
      33             : #include <array>
      34             : #include <cstring>
      35             : #include <type_traits>
      36             : #include <climits>
      37             : #include <limits>
      38             : #include <initializer_list>
      39             : #include <algorithm>
      40             : 
      41             : namespace PLMD {
      42             : 
      43     1868439 : static inline bool typesafePtrSkipCheck() {
      44     1868439 :   static const bool ret=std::getenv("PLUMED_TYPESAFE_IGNORE");
      45     1868439 :   return ret;
      46             : }
      47             : 
      48             : template<class T>
      49     1861841 : std::size_t typesafePtrSizeof() {
      50     1861841 :   return sizeof(T);
      51             : }
      52             : 
      53             : template<>
      54             : inline
      55             : std::size_t typesafePtrSizeof<void>() {
      56             :   return 0;
      57             : }
      58             : 
      59             : template<>
      60             : inline
      61             : std::size_t typesafePtrSizeof<const void>() {
      62             :   return 0;
      63             : }
      64             : 
      65             : /**
      66             : \ingroup TOOLBOX
      67             : Class to deal with propoagation of typesafe pointers.
      68             : 
      69             : */
      70             : class TypesafePtr {
      71             :   /// Small structure used to pass elements of a shape initializer_list
      72             :   struct SizeLike {
      73             :     std::size_t size;
      74             :     SizeLike(short unsigned size): size(size) {}
      75      676192 :     SizeLike(unsigned size): size(size) {}
      76         988 :     SizeLike(long unsigned size): size(size) {}
      77             :     SizeLike(long long unsigned size): size(size) {}
      78             :     SizeLike(short size): size(std::size_t(size)) {}
      79        5380 :     SizeLike(int size): size(std::size_t(size)) {}
      80             :     SizeLike(long int size): size(std::size_t(size)) {}
      81             :     SizeLike(long long int size): size(std::size_t(size)) {}
      82             :   };
      83             : 
      84     1455370 :   inline void init_shape(const std::size_t* shape) {
      85     1455370 :     this->shape[0]=0;
      86     1455370 :     if(shape && *shape>0) {
      87             :       std::size_t nelem_=1;
      88             :       unsigned i=0;
      89     1080755 :       for(i=0; i<this->shape.size(); i++) {
      90     1080755 :         this->shape[i]=*shape;
      91     1080755 :         if(*shape==0) {
      92             :           break;
      93             :         }
      94      673715 :         nelem_*=*shape;
      95      673715 :         shape++;
      96             :       }
      97      407040 :       plumed_assert(i<this->shape.size()); // check that last element is actually zero
      98      407040 :       if(nelem==0) {
      99      406978 :         nelem=nelem_;
     100             :       }
     101      407040 :       plumed_assert(nelem==nelem_) << "Inconsistent shape/nelem";
     102             :     }
     103     1455370 :   }
     104             : 
     105             :   static std::string extra_msg();
     106             : 
     107             : public:
     108             : 
     109             :   TypesafePtr(void* ptr, std::size_t nelem, const std::size_t* shape, std::size_t flags):
     110      421450 :     ptr(ptr),
     111      421450 :     nelem(nelem),
     112      421450 :     flags(flags) {
     113      421450 :     buffer[0]='\0';
     114      421450 :     init_shape(shape);
     115             :   }
     116             : 
     117             :   static const unsigned maxrank=4;
     118             :   static TypesafePtr fromSafePtr(void* safe);
     119             :   static TypesafePtr setNelemAndShape(const TypesafePtr &other, std::size_t nelem, const std::size_t* shape) {
     120      400644 :     return TypesafePtr(other.ptr,nelem,shape,other.flags);
     121             :   }
     122             :   static TypesafePtr unchecked(const void* ptr) {
     123             :     return TypesafePtr(const_cast<void*>(ptr),0,nullptr,0);
     124             :   }
     125             :   static constexpr unsigned short is_integral=3;
     126             :   static constexpr unsigned short is_floating_point=4;
     127             :   static constexpr unsigned short is_file=5;
     128             : 
     129     1825997 :   TypesafePtr() {
     130     1825997 :     shape[0]=0;
     131     1825997 :     buffer[0]='\0';
     132             :   }
     133             : 
     134      296663 :   TypesafePtr(std::nullptr_t) {
     135      296663 :     shape[0]=0;
     136      289707 :     buffer[0]='\0';
     137             :   }
     138             : 
     139             : /// Macro that generate a constructor with given type and flags
     140             : #define __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type,type_,flags_) \
     141             :   TypesafePtr(type_*ptr, std::size_t nelem=0, const std::size_t* shape=nullptr) : \
     142             :     ptr((void*)const_cast<type*>(ptr)), \
     143             :     nelem(nelem), \
     144             :     flags(flags_) \
     145             :   { \
     146             :     init_shape(shape); \
     147             :     buffer[0]='\0'; \
     148             :   }
     149             : 
     150             : /// Macro that uses __PLUMED_WRAPPER_TYPESAFEPTR_INNER to generate constructors with
     151             : /// all possible pointer-const combinations
     152             : #define __PLUMED_WRAPPER_TYPESAFEPTR(type, code,size) \
     153             :   __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type, type,             size | (0x10000*(code)) | (0x2000000*2)) \
     154             :   __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type, type const,       size | (0x10000*(code)) | (0x2000000*3)) \
     155             :   __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type*,type*,            size | (0x10000*(code)) | (0x2000000*4)) \
     156             :   __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type*,type*const,       size | (0x10000*(code)) | (0x2000000*5)) \
     157             :   __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type*,type const*,      size | (0x10000*(code)) | (0x2000000*6)) \
     158             :   __PLUMED_WRAPPER_TYPESAFEPTR_INNER(type*,type const*const, size | (0x10000*(code)) | (0x2000000*7))
     159             : 
     160             : /// Macro that generates the constructors from empy types (those of which sizeof cannot be computed)
     161             : #define __PLUMED_WRAPPER_TYPESAFEPTR_EMPTY(type,code) __PLUMED_WRAPPER_TYPESAFEPTR(type,code,0)
     162             : 
     163             : /// Macro that generates the constructors from sized types (those of which sizeof can be computed).
     164             : /// In addition to generating constructors with all pointer types, it generates a constructor to
     165             : /// allow pass-by-value
     166             : #define __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(type,code) \
     167             :   __PLUMED_WRAPPER_TYPESAFEPTR(type,code,sizeof(type)) \
     168             :   TypesafePtr(type val, std::size_t nelem=0, const std::size_t* shape=nullptr): \
     169             :       nelem(1), \
     170             :       flags(sizeof(type) | (0x10000*(code)) | (0x2000000*1)) \
     171             :     { \
     172             :     plumed_assert(sizeof(type)<=32); \
     173             :     ptr=&buffer[0]; \
     174             :     flags=sizeof(type) | (0x10000*(code)) | (0x2000000*1); \
     175             :     std::memcpy(&buffer[0],&val,sizeof(type)); \
     176             :     init_shape(shape); \
     177             :   }
     178             : 
     179             : /// Here we create all the required instances
     180             : /// 1: void
     181             : /// 3: integral
     182             : /// 4: floating
     183             : /// 5: FILE
     184             : /// 0x100: unsigned
     185        1260 :   __PLUMED_WRAPPER_TYPESAFEPTR_EMPTY(void,1)
     186        2193 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(char,(CHAR_MIN==0)*0x100+3)
     187             :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(signed char,3)
     188             :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(unsigned char,0x100+3)
     189             :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(short,3)
     190             :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(unsigned short,0x100+3)
     191      288606 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(int,3)
     192         102 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(unsigned int,0x100+3)
     193          94 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(long,3)
     194             :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(unsigned long,0x100+3)
     195      267119 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(long long,3)
     196        3939 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(unsigned long long,0x100+3)
     197           0 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(float,4)
     198      339004 :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(double,4)
     199             :   __PLUMED_WRAPPER_TYPESAFEPTR_SIZED(long double,4)
     200        1017 :   __PLUMED_WRAPPER_TYPESAFEPTR_EMPTY(FILE,5)
     201             : 
     202             :   ~TypesafePtr() {
     203     3142739 :   }
     204             : 
     205             : 
     206             :   TypesafePtr(const TypesafePtr&other) = delete;
     207             : 
     208             :   TypesafePtr & operator=(const TypesafePtr & other) = delete;
     209             : 
     210             :   TypesafePtr(TypesafePtr&&other):
     211             :     buffer(other.buffer),
     212             :     ptr(other.ptr==&other.buffer[0] ? &buffer[0] : other.ptr),
     213             :     nelem(other.nelem),
     214             :     shape(other.shape),
     215             :     flags(other.flags) {
     216             :     other.ptr=nullptr;
     217             :   }
     218             : 
     219             :   TypesafePtr copy() const;
     220             : 
     221     1019177 :   TypesafePtr & operator=(TypesafePtr && other) {
     222     1019177 :     ptr=(other.ptr==&other.buffer[0] ? &buffer[0] : other.ptr);
     223     1019177 :     flags=other.flags;
     224     1019177 :     nelem=other.nelem;
     225     1019177 :     shape=other.shape;
     226     1019177 :     buffer=other.buffer;
     227     1019177 :     other.ptr=nullptr;
     228     1019177 :     return *this;
     229             :   }
     230             : 
     231           7 :   std::string type_str() const {
     232           7 :     auto type=(flags>>16)&0xff;
     233           7 :     if(type==0) {
     234           0 :       return "wildcard";
     235             :     }
     236             :     if(type==1) {
     237           2 :       return "void";
     238             :     }
     239             :     if(type==2) {
     240           0 :       return "integral";
     241             :     }
     242             :     if(type==3) {
     243           0 :       return "integral";
     244             :     }
     245             :     if(type==4) {
     246           5 :       return "floating point";
     247             :     }
     248             :     if(type==5) {
     249           0 :       return "FILE";
     250             :     }
     251           0 :     return "unknown";
     252             :   }
     253             : 
     254             : private:
     255             : 
     256             :   template<typename T>
     257     1865235 :   T* get_priv(std::size_t nelem, const std::size_t* shape, bool byvalue) const {
     258             : 
     259     1865235 :     if(typesafePtrSkipCheck()) {
     260           0 :       return (T*) ptr;
     261             :     }
     262             :     typedef typename std::remove_pointer<T>::type T_noptr;
     263     1865235 :     if(flags==0) {
     264         573 :       return (T*) ptr;  // no check
     265             :     }
     266     1862877 :     auto size=flags&0xffff;
     267     1862877 :     auto type=(flags>>16)&0xff;
     268             :     // auto unsi=(flags>>24)&0x1; // ignored
     269     1864662 :     auto cons=(flags>>25)&0x7;
     270             : 
     271             :     // type=0: ignore check
     272             :     // type>5: undefined yet
     273     1862877 :     if(type!=0 && type<=5) {
     274      579894 :       if(std::is_integral<T_noptr>::value && (type!=is_integral)) {
     275          24 :         throw ExceptionTypeError() <<"This command expects an integer type. Received a " << type_str() << " instead"<<extra_msg();
     276             :       }
     277     1281948 :       if(std::is_floating_point<T_noptr>::value && (type!=is_floating_point)) {
     278           4 :         throw ExceptionTypeError() <<"This command expects a floating point type. Received a " << type_str() << " instead"<<extra_msg();
     279             :       }
     280        1034 :       if(std::is_same<FILE,typename std::remove_const<T_noptr>::type>::value && (type!=is_file)) {
     281           0 :         throw ExceptionTypeError() <<"This command expects a FILE. Received a " << type_str() << " instead"<<extra_msg();
     282             :       }
     283             :     }
     284             : 
     285     1862870 :     if(size>0 && typesafePtrSizeof<T_noptr>() >0 && typesafePtrSizeof<T_noptr>()!=size) {
     286          24 :       throw ExceptionTypeError() << "This command expects a type with size " << typesafePtrSizeof<T_noptr>() << ". Received type has size " << size << " instead"<<extra_msg();
     287             :     }
     288             : 
     289     1864649 :     if(!byvalue)
     290     1560391 :       if(cons==1) {
     291           3 :         throw ExceptionTypeError() << "This command is trying to take the address of an argument that was passed by value"<<extra_msg();
     292             :       }
     293             : 
     294             :     // cons==1 (by value) is here treated as cons==3 (const type*)
     295     1859268 :     if(cons>0) {
     296             :       if(!std::is_pointer<T>::value) {
     297             :         if(std::is_void<T>::value) {
     298        1785 :           if(cons==1) {
     299           0 :             throw ExceptionTypeError() << "This command expects a void pointer. It received a value instead"<<extra_msg();
     300             :           }
     301             :         } else {
     302     1857367 :           if(cons!=1 && cons!=2 && cons!=3) {
     303           0 :             throw ExceptionTypeError() << "This command expects a pointer or a value. It received a pointer-to-pointer instead"<<extra_msg();
     304             :           }
     305             :         }
     306             :         if(!std::is_const<T>::value) {
     307      776545 :           if(cons==3) {
     308           6 :             throw ExceptionTypeError() << "This command expects a modifiable pointer (T*). It received a non modifiable pointer instead (const T*)"<<extra_msg();
     309      776543 :           } else if(cons==1) {
     310           0 :             throw ExceptionTypeError() << "This command expects a modifiable pointer (T*). It received a value instead (T)"<<extra_msg();
     311             :           }
     312             :         }
     313             :       } else {
     314             :         if(!std::is_const<T>::value) {
     315         116 :           if(cons==1) {
     316           0 :             throw ExceptionTypeError() << "This command expects a pointer-to-pointer. It received a value intead"<<extra_msg();
     317             :           }
     318         116 :           if(cons==2 || cons==3) {
     319           0 :             throw ExceptionTypeError() << "This command expects a pointer-to-pointer. It received a pointer intead"<<extra_msg();
     320             :           }
     321             :           if(!std::is_const<T_noptr>::value) {
     322             :             if(cons!=4) {
     323             :               throw ExceptionTypeError() << "This command expects a modifiable pointer-to-pointer (T**)"<<extra_msg();
     324             :             }
     325             :           } else {
     326         116 :             if(cons!=6) {
     327           0 :               throw ExceptionTypeError() << "This command expects a modifiable pointer to unmodifiable pointer (const T**)"<<extra_msg();
     328             :             }
     329             :           }
     330             :         } else {
     331             :           if(!std::is_const<T_noptr>::value) {
     332             :             if(cons!=4 && cons!=5) {
     333             :               throw ExceptionTypeError() << "This command expects T*const* pointer, and can only receive T**  or T*const* pointers"<<extra_msg();
     334             :             }
     335             :           }
     336             :         }
     337             :       }
     338             :     }
     339             :     // check full shape, if possible
     340     1864646 :     if(shape && shape[0] && this->shape[0]) {
     341     1361422 :       for(unsigned i=0; i<this->shape.size(); i++) {
     342     1361422 :         if(shape[i]==0 && this->shape[i]!=0) {
     343          12 :           throw ExceptionTypeError() << "Incorrect number of axis (passed greater than requested)"<<extra_msg();
     344             :         }
     345     1361418 :         if(shape[i]!=0 && this->shape[i]==0) {
     346          12 :           throw ExceptionTypeError() << "Incorrect number of axis (requested greater than passed)"<<extra_msg();
     347             :         }
     348     1361414 :         if(shape[i]==0) {
     349             :           break;
     350             :         }
     351      904939 :         if((shape[i]>this->shape[i])) {
     352          80 :           throw ExceptionTypeError() << "This command wants " << shape[i] << " elements on axis " << i <<" of this pointer, but " << this->shape[i] << " have been passed"<<extra_msg();
     353             :         }
     354      904919 :         if(i>0 && (shape[i]<this->shape[i])) {
     355           8 :           throw ExceptionTypeError() << "This command wants " << shape[i] << " elements on axis " << i <<" of this pointer, but " << this->shape[i] << " have been passed"<<extra_msg();
     356             :         }
     357             :       }
     358             :     }
     359     1864616 :     if(nelem==0 && shape && shape[0]>0) {
     360      549033 :       nelem=1;
     361     1580215 :       for(unsigned i=0; i<this->shape.size(); i++) {
     362     1580215 :         if(shape[i]==0) {
     363             :           break;
     364             :         }
     365     1031182 :         nelem*=shape[i];
     366             :       }
     367             :     }
     368             :     // check number of elements
     369     1864616 :     if(nelem>0 && this->nelem>0)
     370      735093 :       if(!(nelem<=this->nelem)) {
     371           8 :         throw ExceptionTypeError() << "This command wants to access " << nelem << " from this pointer, but only " << this->nelem << " have been passed"<<extra_msg();
     372             :       }
     373     1864614 :     return (T*) ptr;
     374             :   }
     375             : 
     376             : public:
     377             : 
     378             :   template<typename T>
     379       13221 :   void set(T val) const {
     380       13221 :     *get_priv<T>(0,nullptr,false)=val;
     381       13221 :   }
     382             : 
     383             :   template<typename T>
     384      858234 :   typename std::enable_if<std::is_pointer<T>::value,T>::type get() const {
     385             :     typedef typename std::remove_pointer<T>::type T_noptr;
     386      861912 :     return get_priv<T_noptr>(0,nullptr,false);
     387             :   }
     388             : 
     389             :   template<typename T>
     390      304338 :   typename std::enable_if<!std::is_pointer<T>::value,T>::type get() const {
     391      304338 :     return *get_priv<const T>(1,nullptr,true);
     392             :   }
     393             : 
     394             :   // this will make sure that a null character is present
     395        3204 :   const char* getCString() const {
     396        3204 :     const char* ptr=get_priv<const char>(0,nullptr,false);
     397        3204 :     if(!typesafePtrSkipCheck() && this->nelem>0 && this->nelem<std::numeric_limits<std::size_t>::max()) {
     398             :       std::size_t i=0;
     399       18736 :       for(; i<this->nelem; i++)
     400       18734 :         if(ptr[i]==0) {
     401             :           break;
     402             :         }
     403         613 :       if(i==this->nelem) {
     404           4 :         throw ExceptionTypeError() << "PLUMED is expecting a null terminated string, but no null character was found";
     405             :       }
     406             :     }
     407        3202 :     return ptr;
     408             :   }
     409             : 
     410             :   template<typename T,typename I>
     411             :   typename std::enable_if<std::is_integral<I>::value, T>::type get(I nelem) const {
     412             :     static_assert(std::is_pointer<T>::value,"only pointer types allowed here");
     413             :     typedef typename std::remove_pointer<T>::type T_noptr;
     414             :     return get_priv<T_noptr>(std::size_t(nelem),nullptr,false);
     415             :   }
     416             : 
     417             :   template<typename T>
     418      682560 :   T get(std::initializer_list<SizeLike> shape) const {
     419             :     static_assert(std::is_pointer<T>::value,"only pointer types allowed here");
     420      682560 :     plumed_assert(shape.size()<=maxrank);
     421             :     std::array<std::size_t,maxrank+1> shape_;
     422             :     typedef typename std::remove_pointer<T>::type T_noptr;
     423             :     unsigned j=0;
     424     1980611 :     for(auto i : shape) {
     425     1298051 :       shape_[j]=i.size;
     426     1298051 :       j++;
     427             :     }
     428      682560 :     shape_[j]=0;
     429      682560 :     return get_priv<T_noptr>(0,&shape_[0],false);
     430             :   }
     431             : 
     432             :   operator bool() const noexcept {
     433      737795 :     return ptr;
     434             :   }
     435             : 
     436             :   void* getRaw() const noexcept {
     437         222 :     return ptr;
     438             :   }
     439             : 
     440             :   std::size_t getNelem() const noexcept {
     441         222 :     return nelem;
     442             :   }
     443             : 
     444             :   const std::size_t* getShape() const noexcept {
     445             :     return shape.data();
     446             :   }
     447             : 
     448             :   std::size_t getFlags() const noexcept {
     449         222 :     return flags;
     450             :   }
     451             : 
     452             : private:
     453             :   std::array<char,32> buffer;
     454             :   void* ptr=nullptr;
     455             :   std::size_t nelem=0;
     456             :   std::array<std::size_t,maxrank+1> shape; // make sure to initialize this!
     457             :   std::size_t flags=0;
     458             : };
     459             : 
     460             : }
     461             : 
     462             : 
     463             : #endif

Generated by: LCOV version 1.16