XTL  0.1
eXtended Template Library
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
mapped_file.hpp
Go to the documentation of this file.
1 
5 #pragma once
6 
7 #include <xtd/xtd.hpp>
8 
9 #if ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
10  #include <sys/mman.h>
11  #include <sys/stat.h>
12  #include <fcntl.h>
13 #endif
14 
15 #include <xtd/memory.hpp>
16 #include <xtd/filesystem.hpp>
17 #include <xtd/exception.hpp>
18 #include <xtd/meta.hpp>
19 
20 namespace xtd{
21 
22  namespace _{
24  template <size_t _PageSize>
26  public:
27  static size_t page_size(){ return _PageSize; }
28  };
29 
31  template <>
32  class mapped_file_base<-1>{
33  public:
34  static size_t page_size(){
35  static size_t iRet = xtd::memory::page_size();
36  return iRet;
37  }
38  };
39  }
40 
41 #if ((XTD_OS_CYGWIN|XTD_OS_MSYS|XTD_OS_LINUX) & XTD_OS)
42 
43 
44  template <typename _Ty> class mapped_page : public std::shared_ptr <_Ty>{
45 
46  using _super_t = std::shared_ptr<_Ty>;
47  template <size_t> friend class mapped_file;
48  explicit mapped_page(void * addr) : _super_t(reinterpret_cast<_Ty*>(addr), [](void*addr) { munmap(addr, 1);}) {}
49  public:
50  template <typename ... _ArgTs> mapped_page(_ArgTs&&...oArgs) : _super_t(std::forward<_ArgTs>(oArgs)...){}
51 
52  mapped_page& operator=(const mapped_page& src){
53  _super_t::operator =(src);
54  return *this;
55  }
56  mapped_page& operator=(mapped_page&& rhs){
57  _super_t::operator =(std::move(rhs));
58  return *this;
59  }
60 
61  void flush(){
62  xtd::crt_exception::throw_if(msync(_super_t::get(), 1, MS_SYNC), [](int i){return 0 != i;});
63  }
64 
65  };
66 
67  template <size_t _page_size>
68  class mapped_file : _::mapped_file_base<_page_size>{
70  int _FileNum;
71  public:
72 
73  ~mapped_file(){ close(_FileNum); }
74 
75  explicit mapped_file(const filesystem::path& Path)
76  : _FileNum(xtd::crt_exception::throw_if(open(Path.string().c_str(), O_CREAT|O_RDWR), [](int i){ return -1==i; }))
77  {}
78 
79 
80 
81 
82  template <typename _Ty> mapped_page<_Ty> get(size_t pageNum){
83  auto iPageSize = _super_t::page_size();
84  struct stat oStat;
85  xtd::crt_exception::throw_if(fstat(_FileNum, &oStat), [](int i){ return -1 == i; });
86  off_t iLastByte = (pageNum * iPageSize) + iPageSize;
87  if (oStat.st_size < iLastByte){
88  xtd::crt_exception::throw_if(lseek(_FileNum, iLastByte, SEEK_SET), [](long l){ return -1==l;});
89  xtd::crt_exception::throw_if(write(_FileNum, "", 1), [](int i){ return 1 != i;});
90  }
91  return mapped_page<_Ty>(
93  mmap(nullptr, iPageSize, PROT_READ|PROT_WRITE, MAP_SHARED, _FileNum, (pageNum * iPageSize)),
94  [](void*addr){ return nullptr==addr || MAP_FAILED==addr; }));
95  }
96 
97  template <typename _Ty> mapped_page<_Ty> append(size_t& newpage){
98  auto iPageSize = _super_t::page_size();
99  struct stat oStat;
100  xtd::crt_exception::throw_if(fstat(_FileNum, &oStat), [](int i){ return -1 == i; });
101  newpage = oStat.st_size / iPageSize;
102  xtd::crt_exception::throw_if(lseek(_FileNum, oStat.st_size + iPageSize, SEEK_SET), [](long l){ return -1==l;});
103  xtd::crt_exception::throw_if(write(_FileNum, "", 1), [](int i){ return 1 != i;});
104 
105  return mapped_page<_Ty>(
107  mmap(nullptr, iPageSize, PROT_READ|PROT_WRITE, MAP_SHARED, _FileNum, oStat.st_size),
108  [](void*addr){ return nullptr==addr || MAP_FAILED==addr; }),
109  [](_Ty*addr){ munmap(addr, _::mapped_file_base<_page_size>::page_size()); });
110  }
111  };
112 #elif ((XTD_OS_MINGW|XTD_OS_WINDOWS) & XTD_OS)
113 
114 
115 
116  template <typename _Ty> class mapped_page : public std::shared_ptr <_Ty>{
117  using _super_t = std::shared_ptr<_Ty>;
118  template <size_t> friend class mapped_file;
119  explicit mapped_page(void * addr) : _super_t(reinterpret_cast<_Ty*>(addr), &UnmapViewOfFile){}
120  public:
121 
122  template <typename ... _ArgTs> mapped_page(_ArgTs&&...oArgs) : _super_t(std::forward<_ArgTs>(oArgs)...){}
123 
124  mapped_page& operator=(const mapped_page& src){
125  _super_t::operator =(src);
126  return *this;
127  }
128  mapped_page& operator=(mapped_page&& rhs){
129  _super_t::operator =(std::move(rhs));
130  return *this;
131  }
132 
133  void flush(){
134  xtd::windows::exception::throw_if(FlushViewOfFile(_super_t::get(), 0), [](BOOL b){return FALSE == b; });
135  }
136 
137  };
138 
139  template <size_t _PageSize>
140  class mapped_file : _::mapped_file_base<_PageSize>{
141  using _super_t = _::mapped_file_base<_PageSize>;
142  HANDLE _hFile;
143  HANDLE _hMap;
144  public:
145  ~mapped_file(){
146  if (_hMap && INVALID_HANDLE_VALUE != _hMap) CloseHandle(_hMap);
147  if (_hFile && INVALID_HANDLE_VALUE != _hFile) CloseHandle(_hFile);
148  }
149  explicit mapped_file(const filesystem::path& Path)
150  : _hFile(xtd::windows::exception::throw_if(CreateFileA(Path.string().c_str(), GENERIC_READ|GENERIC_WRITE, FILE_SHARE_WRITE|FILE_SHARE_READ, nullptr, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, nullptr), [](HANDLE h){ return nullptr==h || INVALID_HANDLE_VALUE==h; }))
151  , _hMap(xtd::windows::exception::throw_if(CreateFileMapping(_hFile, nullptr, PAGE_READWRITE, 0, static_cast<DWORD>(_super_t::page_size()), nullptr), [](HANDLE h){ return nullptr == h || INVALID_HANDLE_VALUE == h; }))
152  {}
153 
154 
155 
156  template <typename _Ty> mapped_page<_Ty> get(size_t pageNum){
157  size_t newSize = (pageNum * _super_t::page_size()) + _super_t::page_size();
158  LARGE_INTEGER iSize;
159  xtd::windows::exception::throw_if(GetFileSizeEx(_hFile, &iSize), [](BOOL b){return FALSE == b; });
160  if (static_cast<size_t>(iSize.QuadPart) < newSize){
161  xtd::windows::exception::throw_if(CloseHandle(_hMap), [](BOOL b){ return FALSE == b; });
162  iSize.QuadPart = newSize;
163  _hMap = xtd::windows::exception::throw_if(CreateFileMapping(_hFile, nullptr, PAGE_READWRITE, iSize.HighPart, iSize.LowPart, nullptr), [](HANDLE h){ return nullptr == h || INVALID_HANDLE_VALUE == h; });
164  }
165  iSize.QuadPart = newSize - _super_t::page_size();
166  return mapped_page<_Ty>(
168  MapViewOfFile(_hMap, FILE_MAP_READ|FILE_MAP_WRITE , iSize.HighPart, iSize.LowPart, _super_t::page_size()),
169  [](void*addr){ return nullptr == addr; }));
170  }
171 
172  template <typename _Ty> mapped_page<_Ty> append(size_t& newpage){
173  LARGE_INTEGER iSize;
174  xtd::windows::exception::throw_if(GetFileSizeEx(_hFile, &iSize), [](BOOL b){return FALSE == b; });
175  newpage = static_cast<size_t>((iSize.QuadPart + _super_t::page_size()) / _super_t::page_size());
176  return get<_Ty>(newpage);
177  }
178 
179  };
180 #endif
181 
182  template<typename _OtherT, typename _ThisT>
183  mapped_page<_OtherT> static_page_cast(_ThisT ptr){
184  return (mapped_page<_OtherT>(ptr, static_cast<_OtherT*>(ptr.get())));
185  }
186 
187 }
188 
static page size
Definition: mapped_file.hpp:25
template meta-programming utilities
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
memory related methods