XTL  0.1
eXtended Template Library
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
dynamic_library.hpp
Go to the documentation of this file.
1 
7 #pragma once
8 
9 #include <xtd/xtd.hpp>
10 
11 #if ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
12  #include <dlfcn.h>
13 #endif
14 
15 #include <memory>
16 #include <map>
17 
18 #include <xtd/exception.hpp>
19 #include <xtd/filesystem.hpp>
20 
21 namespace xtd{
22 
24 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
25  : public xtd::windows::exception{
26  using _super_t = xtd::windows::exception;
27 #elif ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
28  : public xtd::exception{
29  using _super_t = xtd::exception;
30 #endif
31  public:
32 
33  template <typename _ReturnT, typename _ExpressionT>
34  inline static _ReturnT _throw_if(const xtd::source_location& source, _ReturnT ret, _ExpressionT exp, const char* expstr){
35  if (exp(ret)){
36 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
37  throw dynamic_library_exception(source, expstr);
38 #elif ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
39  throw dynamic_library_exception(source, std::string(dlerror()) + " " + expstr);
40 #endif
41  }
42  return ret;
43  }
44 
45  dynamic_library_exception(const source_location& Source, const std::string& What) : _super_t(Source, What){}
48 
49  };
50 
51  class dynamic_library : public std::enable_shared_from_this<dynamic_library>{
52  public:
53  #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
54  using native_handle_type = HMODULE;
55  #elif ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
56  using native_handle_type = void *;
57  #endif
58 
59  using pointer = std::shared_ptr<dynamic_library>;
60  using map = std::map<xtd::filesystem::path, pointer>;
61 
62  static inline pointer make(const char * spath){ return pointer(new dynamic_library(spath)); }
63  static inline pointer make(const xtd::filesystem::path& spath){ return pointer(new dynamic_library(spath.string().c_str())); }
64 
65  native_handle_type handle() const{ return _Handle; }
66 
67  template <typename _ReturnT, typename ... _ArgsT>
68  class function{
69  public:
70  using function_pointer_type = _ReturnT(*)(_ArgsT...);
71 
72  inline _ReturnT operator()(_ArgsT...oArgs) const{
73  return _function_pointer(std::forward<_ArgsT>(oArgs)...);
74  }
75 
76  function() = delete;
77  ~function() = default;
78  function(const function& src) : _function_pointer(src._function_pointer), _library(src._library){}
79  function& operator=(const function& src){
80  if (this==&src){
81  return *this;
82  }
83  _function_pointer = src._function_pointer;
84  _library = src._library;
85  return *this;
86  }
87  private:
88  friend class dynamic_library;
89  function(function_pointer_type fnptr, dynamic_library::pointer oLib) : _function_pointer(fnptr), _library(oLib){}
90  function_pointer_type _function_pointer = nullptr;
91  dynamic_library::pointer _library;
92  };
93 
94  dynamic_library() = delete;
95  dynamic_library(const dynamic_library&) = delete;
96  dynamic_library& operator=(const dynamic_library&) = delete;
97  dynamic_library(dynamic_library&& src) : _Handle(src._Handle){
98  src._Handle = nullptr;
99  }
100  dynamic_library& operator=(dynamic_library&& src){
101  if (this == &src) {
102  return *this;
103  }
104  _Handle = src._Handle;
105  src._Handle = nullptr;
106  return *this;
107  }
108  #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
109  ~dynamic_library(){
110  if (_Handle){
111  FreeLibrary(_Handle);
112  }
113  }
114  #elif ((XTD_OS_LINUX | XTD_OS_CYGWIN) & XTD_OS)
115  ~dynamic_library(){
116  if (_Handle){
117  dlclose(_Handle);
118  }
119  }
120  #endif
121 
122  template <typename _ReturnT, typename ... _ArgsT> function <_ReturnT, _ArgsT...> get(const char * name){
123  using return_type = function <_ReturnT, _ArgsT...>;
124 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
125  auto fnptr = reinterpret_cast<typename return_type::function_pointer_type>(xtd::dynamic_library_exception::throw_if(GetProcAddress(_Handle, name), [](FARPROC p){ return nullptr == p; }));
126 #elif ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
127  auto fnptr = reinterpret_cast<typename return_type::function_pointer_type>(xtd::dynamic_library_exception::throw_if(dlsym(_Handle, name), [](void * p){ return nullptr == p; }));
128 #endif
129  return return_type(fnptr, shared_from_this());
130  }
131 
132  private:
133 
134  friend class process;
135 
136 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
137  explicit dynamic_library(const tchar * sPath) : _Handle(xtd::exception::throw_if(LoadLibrary(sPath), [](HMODULE h){ return (INVALID_HANDLE_VALUE == h || nullptr == h); })){}
138 #elif ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
139  explicit dynamic_library(const char * sPath) : _Handle(xtd::dynamic_library_exception::throw_if(dlopen(sPath, RTLD_LAZY), [](native_handle_type h){ return nullptr == h; })){}
140 #endif
141 
142  explicit dynamic_library(native_handle_type hHandle) : _Handle(hHandle){}
143 
144  native_handle_type _Handle;
145  };
146 }
host, target and build configurations and settings Various components are purpose built for specific ...
#define throw_if(_test, _expression)
Simplifies use of exception::_throw_if.
Definition: exception.hpp:22
generic and special purpose exceptions
handle necessary filesystem and path functionality until C++17 is finalized
Base exception for XTL.
Definition: exception.hpp:27
Contains information about the location of source code Used in error reporting and logging...