ros2_control - rolling
realtime_buffer.hpp
1 // Copyright (c) 2013, hiDOF, Inc.
2 //
3 // Redistribution and use in source and binary forms, with or without
4 // modification, are permitted provided that the following conditions are met:
5 //
6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer.
8 //
9 // * Redistributions in binary form must reproduce the above copyright
10 // notice, this list of conditions and the following disclaimer in the
11 // documentation and/or other materials provided with the distribution.
12 //
13 // * Neither the name of the hiDOF, Inc. nor the names of its
14 // contributors may be used to endorse or promote products derived from
15 // this software without specific prior written permission.
16 //
17 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18 // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21 // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24 // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25 // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26 // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 // POSSIBILITY OF SUCH DAMAGE.
28 
29 /*
30  * Author: Wim Meeussen
31  */
32 
33 #ifndef REALTIME_TOOLS__REALTIME_BUFFER_HPP_
34 #define REALTIME_TOOLS__REALTIME_BUFFER_HPP_
35 
36 #include <chrono>
37 #include <mutex>
38 #include <thread>
39 
40 namespace realtime_tools
41 {
42 template <class T>
44 {
45 public:
46  RealtimeBuffer() : new_data_available_(false)
47  {
48  // allocate memory
49  non_realtime_data_ = new T();
50  realtime_data_ = new T();
51  }
52 
58  explicit RealtimeBuffer(const T & data) : new_data_available_(false)
59  {
60  // allocate memory
61  non_realtime_data_ = new T(data);
62  realtime_data_ = new T(data);
63  }
64 
66  {
67  if (non_realtime_data_) {
68  delete non_realtime_data_;
69  }
70  if (realtime_data_) {
71  delete realtime_data_;
72  }
73  }
74 
75  RealtimeBuffer(const RealtimeBuffer & source)
76  {
77  // allocate memory
78  non_realtime_data_ = new T();
79  realtime_data_ = new T();
80 
81  // Copy the data from old RTB to new RTB
82  writeFromNonRT(*source.readFromNonRT());
83  }
84 
89  {
90  if (this == &source) {
91  return *this;
92  }
93 
94  // Copy the data from old RTB to new RTB
95  writeFromNonRT(*source.readFromNonRT());
96 
97  return *this;
98  }
99 
100  T * readFromRT()
101  {
102  // Check if the data is currently being written to (is locked)
103  std::unique_lock<std::mutex> guard(mutex_, std::try_to_lock);
104  if (guard.owns_lock()) {
105  // swap pointers
106  if (new_data_available_) {
107  T * tmp = realtime_data_;
108  realtime_data_ = non_realtime_data_;
109  non_realtime_data_ = tmp;
110  new_data_available_ = false;
111  }
112  }
113  return realtime_data_;
114  }
115 
116  T * readFromNonRT() const
117  {
118  std::lock_guard<std::mutex> guard(mutex_);
119 
120  if (new_data_available_) {
121  return non_realtime_data_;
122  } else {
123  return realtime_data_;
124  }
125  }
126 
127  void writeFromNonRT(const T & data)
128  {
129 #ifdef NON_POLLING
130  std::lock_guard<std::mutex> guard(mutex_);
131 #else
132  std::unique_lock<std::mutex> guard(mutex_, std::defer_lock);
133  while (!guard.try_lock()) {
134  std::this_thread::sleep_for(std::chrono::microseconds(500));
135  }
136 #endif
137 
138  // copy data into non-realtime buffer
139  *non_realtime_data_ = data;
140  new_data_available_ = true;
141  }
142 
143  void initRT(const T & data)
144  {
145  *non_realtime_data_ = data;
146  *realtime_data_ = data;
147  }
148 
149  void reset()
150  {
151  // delete the old memory
152  if (non_realtime_data_) {
153  delete non_realtime_data_;
154  }
155  if (realtime_data_) {
156  delete realtime_data_;
157  }
158 
159  // allocate memory
160  non_realtime_data_ = new T();
161  realtime_data_ = new T();
162  }
163 
164 private:
165  T * realtime_data_;
166  T * non_realtime_data_;
167  bool new_data_available_;
168 
169  // Set as mutable so that readFromNonRT() can be performed on a const buffer
170  mutable std::mutex mutex_;
171 }; // class
172 
173 } // namespace realtime_tools
174 #endif // REALTIME_TOOLS__REALTIME_BUFFER_HPP_
Definition: realtime_buffer.hpp:44
RealtimeBuffer & operator=(const RealtimeBuffer &source)
Custom assignment operator.
Definition: realtime_buffer.hpp:88
RealtimeBuffer(const T &data)
Constructor for objects that don't have a default constructor.
Definition: realtime_buffer.hpp:58
Definition: async_function_handler.hpp:38