GCC Code Coverage Report


Directory: ./
File: src/shared_memory.cpp
Date: 2022-06-30 06:29:57
Exec Total Coverage
Lines: 62 97 63.9%
Branches: 25 81 30.9%

Line Branch Exec Source
1 /**
2 * @file shared_memory.cpp
3 * @author Maximilien Naveau (maximilien.naveau@gmail.com)
4 * @license License BSD-3-Clause
5 * @copyright Copyright (c) 2019, New York University and Max Planck
6 * Gesellschaft.
7 * @date 2019-05-22
8 *
9 * @brief Implementation of the shared_memory.hpp API
10 */
11 #include "shared_memory/shared_memory.hpp"
12
13 namespace bi = boost::interprocess;
14
15 namespace shared_memory
16 {
17 static uint SEGMENT_SIZE = DEFAULT_SHARED_MEMORY_SIZE;
18 static std::mutex SEGMENT_SIZE_MUTEX;
19
20 9 void set_segment_sizes(uint multiplier_1025)
21 {
22 9 SEGMENT_SIZE_MUTEX.lock();
23 9 SEGMENT_SIZE = multiplier_1025 * 1025;
24 9 SEGMENT_SIZE_MUTEX.unlock();
25 9 }
26
27 1 void set_default_segment_sizes()
28 {
29 1 SEGMENT_SIZE_MUTEX.lock();
30 1 SEGMENT_SIZE = DEFAULT_SHARED_MEMORY_SIZE;
31 1 SEGMENT_SIZE_MUTEX.unlock();
32 1 }
33
34 bool VERBOSE = false;
35 54 void set_verbose(bool mode)
36 {
37 54 VERBOSE = mode;
38 54 }
39
40 static std::chrono::milliseconds time_now()
41 {
42 std::chrono::steady_clock::time_point now =
43 std::chrono::steady_clock::now();
44 std::chrono::milliseconds m =
45 std::chrono::duration_cast<std::chrono::milliseconds>(
46 now.time_since_epoch());
47 return m;
48 }
49
50 bool wait_for_segment(const std::string &segment_id, int timeout_ms)
51 {
52 std::chrono::milliseconds start = time_now();
53 std::chrono::milliseconds timeout{timeout_ms};
54 bool created = false;
55 while (!created)
56 {
57 try
58 {
59 get_segment(segment_id, false, false);
60 return true;
61 }
62 catch (const Non_existing_segment_exception &)
63 {
64 usleep(5);
65 }
66 if (timeout_ms > 0)
67 {
68 if (time_now() - start > timeout)
69 {
70 return false;
71 }
72 }
73 }
74 return true;
75 }
76
77 /***********************************************
78 * Definition of the SharedMemorySegment class *
79 ***********************************************/
80
81 78 SharedMemorySegment::SharedMemorySegment(std::string segment_id,
82 bool clear_upon_destruction,
83 79 bool create)
84 {
85 // save the id the of the segment
86
1/2
✓ Branch 1 taken 78 times.
✗ Branch 2 not taken.
78 segment_id_ = segment_id;
87
88 // check if we should delete the memory upon destruction.
89 78 clear_upon_destruction_ = clear_upon_destruction;
90
91 // create and/or map the memory segment
92
1/2
✓ Branch 1 taken 78 times.
✗ Branch 2 not taken.
78 SEGMENT_SIZE_MUTEX.lock();
93
2/2
✓ Branch 0 taken 77 times.
✓ Branch 1 taken 1 times.
78 if (create)
94 {
95
3/6
✓ Branch 1 taken 77 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 77 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 77 times.
✗ Branch 9 not taken.
154 segment_manager_ = boost::interprocess::managed_shared_memory(
96 boost::interprocess::open_or_create,
97 segment_id.c_str(),
98 77 SEGMENT_SIZE);
99 }
100 else
101 {
102 try
103 {
104
2/6
✗ Branch 2 not taken.
✓ Branch 3 taken 1 times.
✗ Branch 5 not taken.
✗ Branch 6 not taken.
✗ Branch 9 not taken.
✓ Branch 10 taken 1 times.
1 segment_manager_ = boost::interprocess::managed_shared_memory(
105 boost::interprocess::open_only, segment_id.c_str());
106 }
107 2 catch (const boost::interprocess::interprocess_exception &)
108 {
109 1 SEGMENT_SIZE_MUTEX.unlock();
110
1/2
✓ Branch 2 taken 1 times.
✗ Branch 3 not taken.
1 throw Non_existing_segment_exception(segment_id);
111 }
112 }
113 77 SEGMENT_SIZE_MUTEX.unlock();
114
1/2
✓ Branch 1 taken 77 times.
✗ Branch 2 not taken.
77 create_mutex();
115 77 }
116
117 void SharedMemorySegment::clear_memory()
118 {
119 boost::interprocess::shared_memory_object::remove(segment_id_.c_str());
120 }
121
122 3 void SharedMemorySegment::get_object(const std::string &object_id,
123 std::string &get_)
124 {
125 3 mutex_->lock();
126
127 3 register_object_read_only<char>(object_id);
128
129
2/4
✓ Branch 2 taken 3 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 3 times.
✗ Branch 6 not taken.
6 get_ = std::string(static_cast<char *>(objects_[object_id].first),
130
1/2
✓ Branch 1 taken 3 times.
✗ Branch 2 not taken.
3 objects_[object_id].second);
131
132 3 mutex_->unlock();
133 3 }
134
135 18979 SharedMemorySegment &get_segment(const std::string &segment_id,
136 const bool clear_upon_destruction,
137 const bool create)
138 {
139
2/2
✓ Branch 1 taken 78 times.
✓ Branch 2 taken 18901 times.
18979 if (GLOBAL_SHM_SEGMENTS.count(segment_id) == 0)
140 {
141 SharedMemorySegment *s =
142
3/4
✓ Branch 2 taken 78 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 77 times.
✓ Branch 6 taken 1 times.
79 new SharedMemorySegment(segment_id, clear_upon_destruction, create);
143 77 GLOBAL_SHM_SEGMENTS[segment_id].reset(s);
144 }
145 18978 GLOBAL_SHM_SEGMENTS[segment_id]->set_clear_upon_destruction(
146 clear_upon_destruction);
147 18978 return *GLOBAL_SHM_SEGMENTS[segment_id];
148 }
149
150 2 SegmentInfo get_segment_info(const std::string &segment_id)
151 {
152
1/2
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
2 SharedMemorySegment &segment = get_segment(segment_id, false, false);
153
1/2
✓ Branch 1 taken 2 times.
✗ Branch 2 not taken.
2 SegmentInfo si = segment.get_info();
154 2 return si;
155 }
156
157 bool segment_exists(const std::string &segment_id)
158 {
159 if (GLOBAL_SHM_SEGMENTS.count(segment_id) == 0)
160 {
161 return false;
162 }
163 return true;
164 }
165
166 28 boost::interprocess::interprocess_mutex &get_segment_mutex(
167 const std::string segment_id)
168 {
169 28 SharedMemorySegment &segment = get_segment(segment_id);
170 28 return *segment.mutex_;
171 }
172
173 106 void delete_segment(const std::string &segment_id)
174 {
175 // here the unique pointer destroy the object for us.
176 106 GLOBAL_SHM_SEGMENTS.erase(segment_id);
177 106 }
178
179 void delete_all_segment()
180 {
181 delete_all_segments();
182 }
183
184 void delete_all_segments()
185 {
186 for (SegmentMap::iterator seg_it = GLOBAL_SHM_SEGMENTS.begin();
187 seg_it != GLOBAL_SHM_SEGMENTS.end();
188 seg_it = GLOBAL_SHM_SEGMENTS.begin())
189 {
190 get_segment(seg_it->second->get_segment_id(), true);
191 GLOBAL_SHM_SEGMENTS.erase(seg_it);
192 }
193 }
194
195 83 void clear_shared_memory(const std::string &segment_id)
196 {
197 83 boost::interprocess::shared_memory_object::remove(segment_id.c_str());
198 83 delete_segment(segment_id);
199 83 }
200
201 /***********************************
202 * Definition of all set functions *
203 ***********************************/
204
205 3 void set(const std::string &segment_id,
206 const std::string &object_id,
207 const std::string &set_)
208 {
209 3 set<char>(segment_id, object_id, set_.c_str(), set_.size());
210 3 }
211
212 /***********************************
213 * Definition of all get functions *
214 ***********************************/
215
216 3 void get(const std::string &segment_id,
217 const std::string &object_id,
218 std::string &get_,
219 bool create)
220 {
221 try
222 {
223
1/2
✓ Branch 1 taken 3 times.
✗ Branch 2 not taken.
3 SharedMemorySegment &segment = get_segment(segment_id, false, create);
224
1/5
✓ Branch 1 taken 3 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
3 segment.get_object(object_id, get_);
225 }
226 catch (const boost::interprocess::bad_alloc &)
227 {
228 throw shared_memory::Allocation_exception(segment_id, object_id);
229 }
230 catch (const boost::interprocess::interprocess_exception &)
231 {
232 return;
233 }
234 }
235
236
2/4
✓ Branch 1 taken 54 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 54 times.
✗ Branch 4 not taken.
162 } // namespace shared_memory
237