1 //////////////////////////////////////////////////////////////////////////////
2 //
3 // (C) Copyright Ion Gaztanaga 2005-2012. Distributed under the Boost
4 // Software License, Version 1.0. (See accompanying file
5 // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6 //
7 // See http://www.boost.org/libs/interprocess for documentation.
8 //
9 //////////////////////////////////////////////////////////////////////////////
10
11 #ifndef BOOST_INTERPROCESS_SHARED_MEMORY_OBJECT_HPP
12 #define BOOST_INTERPROCESS_SHARED_MEMORY_OBJECT_HPP
13
14 #ifndef BOOST_CONFIG_HPP
15 # include <boost/config.hpp>
16 #endif
17 #
18 #if defined(BOOST_HAS_PRAGMA_ONCE)
19 # pragma once
20 #endif
21
22 #include <boost/interprocess/detail/config_begin.hpp>
23 #include <boost/interprocess/detail/workaround.hpp>
24 #include <boost/interprocess/creation_tags.hpp>
25 #include <boost/interprocess/exceptions.hpp>
26 #include <boost/move/utility_core.hpp>
27 #include <boost/interprocess/interprocess_fwd.hpp>
28 #include <boost/interprocess/exceptions.hpp>
29 #include <boost/interprocess/detail/os_file_functions.hpp>
30 #include <boost/interprocess/detail/shared_dir_helpers.hpp>
31 #include <boost/interprocess/permissions.hpp>
32 #include <boost/move/adl_move_swap.hpp>
33 #include <cstddef>
34 #include <string>
35
36 #if defined(BOOST_INTERPROCESS_POSIX_SHARED_MEMORY_OBJECTS)
37 # include <fcntl.h> //O_CREAT, O_*...
38 # include <sys/mman.h> //shm_xxx
39 # include <unistd.h> //ftruncate, close
40 # include <sys/stat.h> //mode_t, S_IRWXG, S_IRWXO, S_IRWXU,
41 # if defined(BOOST_INTERPROCESS_RUNTIME_FILESYSTEM_BASED_POSIX_SHARED_MEMORY)
42 # if defined(__FreeBSD__)
43 # include <sys/sysctl.h>
44 # endif
45 # endif
46 #else
47 //
48 #endif
49
50 //!\file
51 //!Describes a shared memory object management class.
52
53 namespace boost {
54 namespace interprocess {
55
56 //!A class that wraps a shared memory mapping that can be used to
57 //!create mapped regions from the mapped files
58 class shared_memory_object
59 {
60 #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
61 //Non-copyable and non-assignable
62 BOOST_MOVABLE_BUT_NOT_COPYABLE(shared_memory_object)
63 #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
64
65 public:
66 //!Default constructor. Represents an empty shared_memory_object.
67 shared_memory_object();
68
69 //!Creates a shared memory object with name "name" and mode "mode", with the access mode "mode"
70 //!If the file previously exists, throws an error.*/
shared_memory_object(create_only_t,const char * name,mode_t mode,const permissions & perm=permissions ())71 shared_memory_object(create_only_t, const char *name, mode_t mode, const permissions &perm = permissions())
72 { this->priv_open_or_create(ipcdetail::DoCreate, name, mode, perm); }
73
74 //!Tries to create a shared memory object with name "name" and mode "mode", with the
75 //!access mode "mode". If the file previously exists, it tries to open it with mode "mode".
76 //!Otherwise throws an error.
shared_memory_object(open_or_create_t,const char * name,mode_t mode,const permissions & perm=permissions ())77 shared_memory_object(open_or_create_t, const char *name, mode_t mode, const permissions &perm = permissions())
78 { this->priv_open_or_create(ipcdetail::DoOpenOrCreate, name, mode, perm); }
79
80 //!Tries to open a shared memory object with name "name", with the access mode "mode".
81 //!If the file does not previously exist, it throws an error.
shared_memory_object(open_only_t,const char * name,mode_t mode)82 shared_memory_object(open_only_t, const char *name, mode_t mode)
83 { this->priv_open_or_create(ipcdetail::DoOpen, name, mode, permissions()); }
84
85 //!Moves the ownership of "moved"'s shared memory object to *this.
86 //!After the call, "moved" does not represent any shared memory object.
87 //!Does not throw
shared_memory_object(BOOST_RV_REF (shared_memory_object)moved)88 shared_memory_object(BOOST_RV_REF(shared_memory_object) moved)
89 : m_handle(file_handle_t(ipcdetail::invalid_file()))
90 , m_mode(read_only)
91 { this->swap(moved); }
92
93 //!Moves the ownership of "moved"'s shared memory to *this.
94 //!After the call, "moved" does not represent any shared memory.
95 //!Does not throw
operator =(BOOST_RV_REF (shared_memory_object)moved)96 shared_memory_object &operator=(BOOST_RV_REF(shared_memory_object) moved)
97 {
98 shared_memory_object tmp(boost::move(moved));
99 this->swap(tmp);
100 return *this;
101 }
102
103 //!Swaps the shared_memory_objects. Does not throw
104 void swap(shared_memory_object &moved);
105
106 //!Erases a shared memory object from the system.
107 //!Returns false on error. Never throws
108 static bool remove(const char *name);
109
110 //!Sets the size of the shared memory mapping
111 void truncate(offset_t length);
112
113 //!Destroys *this and indicates that the calling process is finished using
114 //!the resource. All mapped regions are still
115 //!valid after destruction. The destructor function will deallocate
116 //!any system resources allocated by the system for use by this process for
117 //!this resource. The resource can still be opened again calling
118 //!the open constructor overload. To erase the resource from the system
119 //!use remove().
120 ~shared_memory_object();
121
122 //!Returns the name of the shared memory object.
123 const char *get_name() const;
124
125 //!Returns true if the size of the shared memory object
126 //!can be obtained and writes the size in the passed reference
127 bool get_size(offset_t &size) const;
128
129 //!Returns access mode
130 mode_t get_mode() const;
131
132 //!Returns mapping handle. Never throws.
133 mapping_handle_t get_mapping_handle() const;
134
135 #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
136 private:
137
138 //!Closes a previously opened file mapping. Never throws.
139 void priv_close();
140
141 //!Opens or creates a shared memory object.
142 bool priv_open_or_create(ipcdetail::create_enum_t type, const char *filename, mode_t mode, const permissions &perm);
143
144 file_handle_t m_handle;
145 mode_t m_mode;
146 std::string m_filename;
147 #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
148 };
149
150 #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
151
shared_memory_object()152 inline shared_memory_object::shared_memory_object()
153 : m_handle(file_handle_t(ipcdetail::invalid_file()))
154 , m_mode(read_only)
155 {}
156
~shared_memory_object()157 inline shared_memory_object::~shared_memory_object()
158 { this->priv_close(); }
159
160
get_name() const161 inline const char *shared_memory_object::get_name() const
162 { return m_filename.c_str(); }
163
get_size(offset_t & size) const164 inline bool shared_memory_object::get_size(offset_t &size) const
165 { return ipcdetail::get_file_size((file_handle_t)m_handle, size); }
166
swap(shared_memory_object & other)167 inline void shared_memory_object::swap(shared_memory_object &other)
168 {
169 boost::adl_move_swap(m_handle, other.m_handle);
170 boost::adl_move_swap(m_mode, other.m_mode);
171 m_filename.swap(other.m_filename);
172 }
173
get_mapping_handle() const174 inline mapping_handle_t shared_memory_object::get_mapping_handle() const
175 {
176 return ipcdetail::mapping_handle_from_file_handle(m_handle);
177 }
178
get_mode() const179 inline mode_t shared_memory_object::get_mode() const
180 { return m_mode; }
181
182 #if !defined(BOOST_INTERPROCESS_POSIX_SHARED_MEMORY_OBJECTS)
183
priv_open_or_create(ipcdetail::create_enum_t type,const char * filename,mode_t mode,const permissions & perm)184 inline bool shared_memory_object::priv_open_or_create
185 (ipcdetail::create_enum_t type, const char *filename, mode_t mode, const permissions &perm)
186 {
187 m_filename = filename;
188 std::string shmfile;
189 ipcdetail::create_shared_dir_cleaning_old_and_get_filepath(filename, shmfile);
190
191 //Set accesses
192 if (mode != read_write && mode != read_only){
193 error_info err = other_error;
194 throw interprocess_exception(err);
195 }
196
197 switch(type){
198 case ipcdetail::DoOpen:
199 m_handle = ipcdetail::open_existing_file(shmfile.c_str(), mode, true);
200 break;
201 case ipcdetail::DoCreate:
202 m_handle = ipcdetail::create_new_file(shmfile.c_str(), mode, perm, true);
203 break;
204 case ipcdetail::DoOpenOrCreate:
205 m_handle = ipcdetail::create_or_open_file(shmfile.c_str(), mode, perm, true);
206 break;
207 default:
208 {
209 error_info err = other_error;
210 throw interprocess_exception(err);
211 }
212 }
213
214 //Check for error
215 if(m_handle == ipcdetail::invalid_file()){
216 error_info err = system_error_code();
217 this->priv_close();
218 throw interprocess_exception(err);
219 }
220
221 m_mode = mode;
222 return true;
223 }
224
remove(const char * filename)225 inline bool shared_memory_object::remove(const char *filename)
226 {
227 try{
228 //Make sure a temporary path is created for shared memory
229 std::string shmfile;
230 ipcdetail::shared_filepath(filename, shmfile);
231 return ipcdetail::delete_file(shmfile.c_str());
232 }
233 catch(...){
234 return false;
235 }
236 }
237
truncate(offset_t length)238 inline void shared_memory_object::truncate(offset_t length)
239 {
240 if(!ipcdetail::truncate_file(m_handle, length)){
241 error_info err = system_error_code();
242 throw interprocess_exception(err);
243 }
244 }
245
priv_close()246 inline void shared_memory_object::priv_close()
247 {
248 if(m_handle != ipcdetail::invalid_file()){
249 ipcdetail::close_file(m_handle);
250 m_handle = ipcdetail::invalid_file();
251 }
252 }
253
254 #else //!defined(BOOST_INTERPROCESS_POSIX_SHARED_MEMORY_OBJECTS)
255
256 namespace shared_memory_object_detail {
257
258 #ifdef BOOST_INTERPROCESS_RUNTIME_FILESYSTEM_BASED_POSIX_SHARED_MEMORY
259
260 #if defined(__FreeBSD__)
261
use_filesystem_based_posix()262 inline bool use_filesystem_based_posix()
263 {
264 int jailed = 0;
265 std::size_t len = sizeof(jailed);
266 ::sysctlbyname("security.jail.jailed", &jailed, &len, NULL, 0);
267 return jailed != 0;
268 }
269
270 #else
271 #error "Not supported platform for BOOST_INTERPROCESS_RUNTIME_FILESYSTEM_BASED_POSIX_SHARED_MEMORY"
272 #endif
273
274 #endif
275
276 } //shared_memory_object_detail
277
priv_open_or_create(ipcdetail::create_enum_t type,const char * filename,mode_t mode,const permissions & perm)278 inline bool shared_memory_object::priv_open_or_create
279 (ipcdetail::create_enum_t type,
280 const char *filename,
281 mode_t mode, const permissions &perm)
282 {
283 #if defined(BOOST_INTERPROCESS_FILESYSTEM_BASED_POSIX_SHARED_MEMORY)
284 const bool add_leading_slash = false;
285 #elif defined(BOOST_INTERPROCESS_RUNTIME_FILESYSTEM_BASED_POSIX_SHARED_MEMORY)
286 const bool add_leading_slash = !shared_memory_object_detail::use_filesystem_based_posix();
287 #else
288 const bool add_leading_slash = true;
289 #endif
290 if(add_leading_slash){
291 ipcdetail::add_leading_slash(filename, m_filename);
292 }
293 else{
294 ipcdetail::create_shared_dir_cleaning_old_and_get_filepath(filename, m_filename);
295 }
296
297 //Create new mapping
298 int oflag = 0;
299 if(mode == read_only){
300 oflag |= O_RDONLY;
301 }
302 else if(mode == read_write){
303 oflag |= O_RDWR;
304 }
305 else{
306 error_info err(mode_error);
307 throw interprocess_exception(err);
308 }
309 int unix_perm = perm.get_permissions();
310
311 switch(type){
312 case ipcdetail::DoOpen:
313 {
314 //No oflag addition
315 m_handle = shm_open(m_filename.c_str(), oflag, unix_perm);
316 }
317 break;
318 case ipcdetail::DoCreate:
319 {
320 oflag |= (O_CREAT | O_EXCL);
321 m_handle = shm_open(m_filename.c_str(), oflag, unix_perm);
322 if(m_handle >= 0){
323 ::fchmod(m_handle, unix_perm);
324 }
325 }
326 break;
327 case ipcdetail::DoOpenOrCreate:
328 {
329 //We need a create/open loop to change permissions correctly using fchmod, since
330 //with "O_CREAT" only we don't know if we've created or opened the shm.
331 while(true){
332 //Try to create shared memory
333 m_handle = shm_open(m_filename.c_str(), oflag | (O_CREAT | O_EXCL), unix_perm);
334 //If successful change real permissions
335 if(m_handle >= 0){
336 ::fchmod(m_handle, unix_perm);
337 }
338 //If already exists, try to open
339 else if(errno == EEXIST){
340 m_handle = shm_open(m_filename.c_str(), oflag, unix_perm);
341 //If open fails and errno tells the file does not exist
342 //(shm was removed between creation and opening tries), just retry
343 if(m_handle < 0 && errno == ENOENT){
344 continue;
345 }
346 }
347 //Exit retries
348 break;
349 }
350 }
351 break;
352 default:
353 {
354 error_info err = other_error;
355 throw interprocess_exception(err);
356 }
357 }
358
359 //Check for error
360 if(m_handle < 0){
361 error_info err = errno;
362 this->priv_close();
363 throw interprocess_exception(err);
364 }
365
366 m_filename = filename;
367 m_mode = mode;
368 return true;
369 }
370
remove(const char * filename)371 inline bool shared_memory_object::remove(const char *filename)
372 {
373 try{
374 std::string filepath;
375 #if defined(BOOST_INTERPROCESS_FILESYSTEM_BASED_POSIX_SHARED_MEMORY)
376 const bool add_leading_slash = false;
377 #elif defined(BOOST_INTERPROCESS_RUNTIME_FILESYSTEM_BASED_POSIX_SHARED_MEMORY)
378 const bool add_leading_slash = !shared_memory_object_detail::use_filesystem_based_posix();
379 #else
380 const bool add_leading_slash = true;
381 #endif
382 if(add_leading_slash){
383 ipcdetail::add_leading_slash(filename, filepath);
384 }
385 else{
386 ipcdetail::shared_filepath(filename, filepath);
387 }
388 return 0 == shm_unlink(filepath.c_str());
389 }
390 catch(...){
391 return false;
392 }
393 }
394
truncate(offset_t length)395 inline void shared_memory_object::truncate(offset_t length)
396 {
397 if(0 != ftruncate(m_handle, length)){
398 error_info err(system_error_code());
399 throw interprocess_exception(err);
400 }
401 }
402
priv_close()403 inline void shared_memory_object::priv_close()
404 {
405 if(m_handle != -1){
406 ::close(m_handle);
407 m_handle = -1;
408 }
409 }
410
411 #endif
412
413 //!A class that stores the name of a shared memory
414 //!and calls shared_memory_object::remove(name) in its destructor
415 //!Useful to remove temporary shared memory objects in the presence
416 //!of exceptions
417 class remove_shared_memory_on_destroy
418 {
419 const char * m_name;
420 public:
remove_shared_memory_on_destroy(const char * name)421 remove_shared_memory_on_destroy(const char *name)
422 : m_name(name)
423 {}
424
~remove_shared_memory_on_destroy()425 ~remove_shared_memory_on_destroy()
426 { shared_memory_object::remove(m_name); }
427 };
428
429 #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
430
431 } //namespace interprocess {
432 } //namespace boost {
433
434 #include <boost/interprocess/detail/config_end.hpp>
435
436 #endif //BOOST_INTERPROCESS_SHARED_MEMORY_OBJECT_HPP
437