TAMSVIZ
Visualization and annotation tool for ROS
type.h
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #pragma once
5 
6 #include <functional>
7 #include <memory>
8 #include <stdexcept>
9 #include <typeindex>
10 #include <vector>
11 
12 inline void toString(const std::string &v, std::string &s) { s = v; }
13 inline void toString(const bool &v, std::string &s) {
14  if (v) {
15  s = "true";
16  } else {
17  s = "false";
18  }
19 }
20 inline void toString(float v, std::string &s) { s = std::to_string(v); }
21 inline void toString(double v, std::string &s) { s = std::to_string(v); }
22 inline void toString(int v, std::string &s) { s = std::to_string(v); }
23 inline void toString(long v, std::string &s) { s = std::to_string(v); }
24 inline void toString(long long v, std::string &s) { s = std::to_string(v); }
25 inline void toString(unsigned long v, std::string &s) { s = std::to_string(v); }
26 inline void toString(unsigned long long v, std::string &s) {
27  s = std::to_string(v);
28 }
29 
30 inline void fromString(std::string &v, const std::string &s) { v = s; }
31 inline void fromString(float &v, const std::string &s) { v = std::stof(s); }
32 inline void fromString(double &v, const std::string &s) { v = std::stod(s); }
33 inline void fromString(int &v, const std::string &s) { v = std::stoi(s); }
34 inline void fromString(long &v, const std::string &s) { v = std::stol(s); }
35 inline void fromString(long long &v, const std::string &s) {
36  v = std::stoll(s);
37 }
38 inline void fromString(unsigned long &v, const std::string &s) {
39  v = std::stoul(s);
40 }
41 inline void fromString(unsigned long long &v, const std::string &s) {
42  v = std::stoull(s);
43 }
44 inline void fromString(bool &v, const std::string &s) {
45  if (s == "true" || s == "1" || s == "True") {
46  v = true;
47  return;
48  }
49  if (s == "false" || s == "0" || s == "False") {
50  v = false;
51  return;
52  }
53  throw std::invalid_argument("s");
54 }
55 
56 class SnapshotBase;
57 class Object;
58 class Property;
59 
60 static inline bool tryToString(...) { return false; }
61 template <class T>
62 static inline auto tryToString(const T *v, std::string &s)
63  -> decltype(toString(*v, s), bool()) {
64  toString(*v, s);
65  return true;
66 }
67 
68 static inline void fromStringOrThrow(...) {
69  throw std::invalid_argument("type");
70 }
71 template <class T>
72 static inline auto fromStringOrThrow(T *v, const std::string &s)
73  -> decltype(fromString(*v, s)) {
74  fromString(*v, s);
75 }
76 
77 static constexpr bool toStringSupported(...) { return false; }
78 template <class T>
79 static constexpr auto toStringSupported(T *v)
80  -> decltype(toString(*v, *(std::string *)nullptr), true) {
81  return true;
82 }
83 
84 static constexpr bool fromStringSupported(...) { return false; }
85 template <class T>
86 static constexpr auto fromStringSupported(T *v)
87  -> decltype(fromString(*v, *(std::string *)nullptr), true) {
88  return true;
89 }
90 
91 static constexpr bool propertiesSupported(...) { return false; }
92 template <class T>
93 static constexpr auto propertiesSupported(T *v)
94  -> decltype(properties(*v, *(std::vector<Property> *)nullptr), true) {
95  return true;
96 }
97 
98 inline void removeObject(...) {}
99 template <class T>
100 void removeObject(std::vector<std::shared_ptr<T>> &parent, void *object) {
101  for (auto it = parent.begin(); it != parent.end();) {
102  if (it->get() == object) {
103  it = parent.erase(it);
104  } else {
105  removeObject(*it, object);
106  it++;
107  }
108  }
109 }
110 void removeObject(const std::shared_ptr<Object> &parent, void *object);
111 
113 
114 typedef std::function<bool(const std::shared_ptr<Object> &o)> SnapshotFilter;
115 
116 uint64_t handleObjectId(const Object *object);
117 template <class T> class Handle {
118  uint64_t _id = 0;
119 
120 public:
121  Handle(const std::shared_ptr<T> &p = nullptr) { reset(p); }
122  void reset(uint64_t id) { _id = id; }
123  void reset(const std::shared_ptr<T> &p = nullptr) {
124  _id = handleObjectId(p ? (Object *)p.get() : nullptr);
125  }
126  void operator=(const std::shared_ptr<T> &p) { reset(p); }
127  template <class R>
128  std::shared_ptr<T> resolve(const std::shared_ptr<R> &root) {
129  if (_id == 0) {
130  return nullptr;
131  }
132  std::shared_ptr<T> r;
133  root->recurse([&r, this](const std::shared_ptr<Object> &o) {
134  if (handleObjectId(o.get()) == _id) {
135  if (auto x = std::dynamic_pointer_cast<T>(o)) {
136  r = x;
137  }
138  }
139  });
140  return r;
141  }
142  const uint64_t id() const { return _id; }
143 };
144 template <class T, class S>
145 bool operator==(const Handle<T> &h, const std::shared_ptr<S> &p) {
146  if (p == nullptr) {
147  return h.id() == 0;
148  } else {
149  return h.id() == p->id();
150  }
151 }
152 template <class T, class S>
153 bool operator==(const std::shared_ptr<S> &p, const Handle<T> &h) {
154  if (p == nullptr) {
155  return h.id() == 0;
156  } else {
157  return h.id() == p->id();
158  }
159 }
160 template <class T, class S>
161 bool operator!=(const Handle<T> &h, const std::shared_ptr<S> &p) {
162  return !(h == p);
163 }
164 template <class T, class S>
165 bool operator!=(const std::shared_ptr<S> &p, const Handle<T> &h) {
166  return !(h == p);
167 }
168 
169 template <class T>
170 auto forEachObject(void *context,
171  void (*f)(void *, const Object *, const Object *),
172  const Object *parent, const T &child) ->
173  typename std::enable_if<propertiesSupported((T *)nullptr) ||
174  fromStringSupported((T *)nullptr),
175  StopObjectRecursionTag>::type {
176  return StopObjectRecursionTag();
177 }
178 
179 void forEachObject(void *context,
180  void (*f)(void *, const Object *, const Object *),
181  const Object *parent, const Object *child);
182 
183 template <class T>
184 void forEachObject(void *context,
185  void (*f)(void *, const Object *, const Object *),
186  const Object *parent, const std::shared_ptr<T> &v) {
187  forEachObject(context, f, parent, v.get());
188 }
189 
190 template <class T>
191 void forEachObject(void *context,
192  void (*f)(void *, const Object *, const Object *),
193  const Object *parent, const std::vector<T> &v) {
194  for (auto &e : v) {
195  forEachObject(context, f, parent, e);
196  }
197 }
198 
199 template <class T>
201 forEachObject(void *context, void (*f)(void *, const Object *, const Object *),
202  const Object *parent, const Handle<T> &v) {
203  return StopObjectRecursionTag();
204 }
205 
206 static void tryGetProperties(...) {}
207 template <class X>
208 static auto tryGetProperties(X *v, std::vector<Property> &props)
209  -> decltype(properties(*v, props)) {
210  properties(*v, props);
211 }
212 void getObjectProperties(const std::shared_ptr<Object> &v,
213  std::vector<Property> &props);
214 template <class O>
215 static void tryGetProperties(std::shared_ptr<O> *v,
216  std::vector<Property> &props) {
217  getObjectProperties(*v, props);
218 }
219 
220 struct Type {
221 protected:
222  std::string _name;
223  std::type_index _type_id = typeid(void);
224  std::shared_ptr<Type> _base;
225  bool _constructable = false;
226  std::function<std::shared_ptr<void>()> _create;
227  std::string _category;
228 
229  static void _registerType(const std::shared_ptr<Type> &t);
230 
231  template <class T> void _initConstructor(...) {
232  _create = []() -> std::shared_ptr<void> {
233  throw std::runtime_error(
234  "can't instantiate class without default constructor");
235  };
236  }
237  template <class T> void _initConstructor(decltype(new T()) i) {
238  _create = []() -> std::shared_ptr<void> {
239  return std::static_pointer_cast<void>(std::make_shared<T>());
240  };
241  _constructable = true;
242  }
243 
244  Type() {}
245  Type(const Type &) = delete;
246  Type &operator=(const Type &) = delete;
247  virtual ~Type() {}
248 
249 public:
250  const std::type_index &typeId() const { return _type_id; }
251  const std::string &name() const { return _name; }
252  const std::shared_ptr<Type> &base() const { return _base; }
253  bool constructable() const { return _constructable; }
254  const std::string &category() const { return _category; }
255 
256  template <class T> std::shared_ptr<T> instantiate() const {
257  for (const Type *t = this; t; t = t->_base.get()) {
258  if (t->typeId() == typeid(T)) {
259  return std::static_pointer_cast<T>(_create());
260  }
261  }
262  return nullptr;
263  }
264 
265  static std::shared_ptr<Type> tryFind(const std::string &name);
266  static std::shared_ptr<Type> tryFind(const std::type_index &id);
267  static std::shared_ptr<Type> find(const std::string &name);
268  static std::shared_ptr<Type> find(const std::type_index &id);
269  template <class T> static std::shared_ptr<Type> find() {
270  return find(typeid(T));
271  }
272  template <class T> static std::shared_ptr<Type> tryFind() {
273  return tryFind(typeid(T));
274  }
275 
276  std::vector<std::shared_ptr<Type>> list() const;
277 
278  template <class T>
279  static std::shared_ptr<Type>
280  global(const std::string &name = "",
281  const std::shared_ptr<Type> &base = nullptr,
282  const std::string &category = "") {
283  if (auto ret = tryFind<T>()) {
284  return ret;
285  }
286  auto ret = create<T>(name, base, category);
287  _registerType(ret);
288  return ret;
289  }
290 
291  template <class T>
292  static std::shared_ptr<Type>
293  create(const std::string &name = "",
294  const std::shared_ptr<Type> &base = nullptr,
295  const std::string &category = "");
296 
297  virtual bool tryToString(const void *p, std::string &s) const = 0;
298  virtual bool fromStringSupported() const = 0;
299  virtual void fromStringOrThrow(void *p, const std::string &s) const = 0;
300  virtual void properties(const void *ptr,
301  std::vector<Property> &properties) const = 0;
302 
303 protected:
304  static bool _tryToPointerList(...) { return false; }
305  template <class T>
306  static bool _tryToPointerList(const std::vector<std::shared_ptr<T>> *list,
307  std::vector<std::shared_ptr<void>> &out) {
308  out.assign(list->begin(), list->end());
309  return true;
310  }
311 
312  static bool _tryFromPointerList(...) { return false; }
313  template <class T>
314  static bool
315  _tryFromPointerList(std::vector<std::shared_ptr<T>> *list,
316  const std::vector<std::shared_ptr<void>> &in) {
317  list->clear();
318  list->reserve(in.size());
319  for (auto &v : in) {
320  list->push_back(std::static_pointer_cast<T>(v));
321  }
322  return true;
323  }
324 
325 public:
326  virtual bool
327  tryToPointerList(const void *object,
328  std::vector<std::shared_ptr<void>> &out) const = 0;
329  virtual bool
330  tryFromPointerList(void *object,
331  const std::vector<std::shared_ptr<void>> &in) const = 0;
332 };
333 
334 template <class T> class TypeImpl : public Type {
335 
336 public:
337  TypeImpl(const std::string &name, const std::shared_ptr<Type> &base,
338  const std::string &category) {
339  _name = name;
340  _base = base;
341  _category = category;
342  _type_id = typeid(T);
343  _initConstructor<T>(0);
344  }
345  virtual bool tryToString(const void *p, std::string &s) const override {
346  return ::tryToString((const T *)p, s);
347  }
348  virtual bool fromStringSupported() const {
349  return ::fromStringSupported((T *)nullptr);
350  }
351  virtual void fromStringOrThrow(void *value,
352  const std::string &s) const override {
353  ::fromStringOrThrow((T *)value, s);
354  }
355  virtual void properties(const void *ptr,
356  std::vector<Property> &props) const override {
357  tryGetProperties((T *)ptr, props);
358  }
359  virtual bool
360  tryToPointerList(const void *object,
361  std::vector<std::shared_ptr<void>> &out) const override {
362  return _tryToPointerList(static_cast<const T *>(object), out);
363  }
364  virtual bool tryFromPointerList(
365  void *object,
366  const std::vector<std::shared_ptr<void>> &in) const override {
367  return _tryFromPointerList(static_cast<T *>(object), in);
368  }
369 };
370 
371 template <class T>
372 std::shared_ptr<Type> Type::create(const std::string &name,
373  const std::shared_ptr<Type> &base,
374  const std::string &category) {
375  return std::shared_ptr<Type>(new TypeImpl<T>(name, base, category));
376 }
377 
378 #define DECLARE_TYPE_STRINGIFY_2(x) #x
379 #define DECLARE_TYPE_STRINGIFY(x) DECLARE_TYPE_STRINGIFY_2(x)
380 
381 #define DECLARE_TYPE_IMPL(Derived, Base, ...) \
382  static_assert(std::is_base_of<Base, Derived>::value, \
383  "incorrect base class"); \
384  static std::shared_ptr<Type> _##Base##_##Derived = Type::global<Derived>( \
385  #Derived, \
386  (typeid(Base) != typeid(Derived)) ? Type::find(typeid(Base)) : nullptr);
387 #define DECLARE_TYPE(Type, ...) DECLARE_TYPE_IMPL(Type, ##__VA_ARGS__, Type)
388 
389 #define DECLARE_TYPE_C(Derived, Base, Category) \
390  static_assert(std::is_base_of<Base, Derived>::value, \
391  "incorrect base class"); \
392  static std::shared_ptr<Type> _##Base##_##Derived = Type::global<Derived>( \
393  #Derived, Type::find(typeid(Base)), DECLARE_TYPE_STRINGIFY_2(Category));
Definition: object.h:8
Definition: type.h:117
Definition: type.h:220
Definition: type.h:334