Graphite Version 3
An experimental 3D geometry processing program
Loading...
Searching...
No Matches
any.h
Go to the documentation of this file.
1/*
2 * OGF/Graphite: Geometry and Graphics Programming Library + Utilities
3 * Copyright (C) 2000 Bruno Levy
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 *
19 * If you modify this software, you should include a notice giving the
20 * name of the person performing the modification, the date of modification,
21 * and the reason for such modification.
22 *
23 * Contact: Bruno Levy
24 *
25 * levy@loria.fr
26 *
27 * ISA Project
28 * LORIA, INRIA Lorraine,
29 * Campus Scientifique, BP 239
30 * 54506 VANDOEUVRE LES NANCY CEDEX
31 * FRANCE
32 *
33 * Note that the GNU General Public License does not permit incorporating
34 * the Software into proprietary programs.
35 */
36
37#ifndef H_OGF_BASIC_TYPES_ANY_H
38#define H_OGF_BASIC_TYPES_ANY_H
39
41#include <OGF/gom/services/life_cycle.h>
42#include <typeinfo>
43
49#ifdef GEO_COMPILER_CLANG
50#pragma GCC diagnostic push
51#pragma GCC diagnostic ignored "-Wcast-align"
52#endif
53
54
55namespace OGF {
56
57 class MetaType;
58
62 class GOM_API Any {
63 private:
64
69 enum { BUFFER_SIZE = 40 };
70
71 public:
72
76 Any() : value_(nullptr), in_buffer_(false), meta_type_(nullptr) {
77 }
78
82 ~Any() {
83 destroy();
84 }
85
91 bool is_null() const {
92 return value_ == nullptr;
93 }
94
98 void reset() {
99 destroy();
100 }
101
106 Any(const Any& rhs) :
107 value_(nullptr), in_buffer_(false), meta_type_(nullptr) {
108 copy(rhs);
109 }
110
116 Any& operator=(const Any& rhs) {
117 if(&rhs != this) {
118 destroy();
119 copy(rhs);
120 }
121 return *this;
122 }
123
128 std::string as_string() const {
129 std::string result;
130 if(value_ != nullptr) {
131 convert_to_string(meta_type_, result, value_);
132 }
133 return result;
134 }
135
142 return meta_type_;
143 }
144
152
157 template <class T> void set_value(const T& value) {
158 MetaType* new_type = resolve_meta_type<T>();
159 if(new_type == meta_type_) {
160 life_cycle()->assign(value_, Memory::pointer(&value));
161 return;
162 }
163 destroy();
164 meta_type_ = new_type;
165
166#if defined(GEO_COMPILER_GCC)
167// GCC does not see that we use placement new in the buffer
168// only if object size is smaller than BUFFER_SIZE and
169// generates a warning.
170#pragma GCC diagnostic push
171#pragma GCC diagnostic ignored "-Wplacement-new"
172#elif defined(GEO_COMPILER_MSVC)
173// MSVC barks on test with constant condition.
174#pragma warning(push)
175#pragma warning(disable:4127)
176#endif
177
178 if(sizeof(T) <= BUFFER_SIZE) {
179 in_buffer_ = true;
180 value_ = buffer_;
181 new(value_)T(value);
182 } else {
183 in_buffer_ = false;
184 value_ = Memory::pointer(new T(value));
185 }
186
187#if defined(GEO_COMPILER_GCC)
188#pragma GCC diagnostic pop
189#elif defined(GEO_COMPILER_MSVC)
190#pragma warning(pop)
191#endif
192
193 }
194
202 void set_value(const char* value) {
203 set_value<std::string>(std::string(value));
204 }
205
212 template <class T> bool get_value(T& value) const {
213 if(value_ == nullptr) {
214 return false;
215 }
216 // Test if the stored value is of type T
217 if(meta_type_ == resolve_meta_type<T>()) {
218 value = value_as<T>();
219 return true;
220 }
221 // If the stored value is a string, convert it to T
222 if(meta_type_ == resolve_meta_type<std::string>()) {
223 const std::string& string_value = value_as<std::string>();
224 convert_from_string(
225 resolve_meta_type<T>(),
226 string_value, (Memory::pointer)(&value)
227 );
228 return true;
229 }
230 return false;
231 }
232
239 bool get_value(std::string& value) const {
240 value = as_string();
241 return true;
242 }
243
250 bool get_value(index_t& value) const {
251 if(get_value<index_t>(value)) {
252 return true;
253 }
254 {
255 signed_index_t tmp;
256 if(get_value<signed_index_t>(tmp)) {
257 value = index_t(tmp);
258 return true;
259 }
260 }
261 {
262 float tmp;
263 if(get_value<float>(tmp)) {
264 value = index_t(tmp);
265 return true;
266 }
267
268 }
269 {
270 double tmp;
271 if(get_value<double>(tmp)) {
272 value = index_t(tmp);
273 return true;
274 }
275
276 }
277 return false;
278 }
279
286 bool get_value(signed_index_t& value) const {
287 if(get_value<signed_index_t>(value)) {
288 return true;
289 }
290 {
291 index_t tmp;
292 if(get_value<index_t>(tmp)) {
293 value = signed_index_t(tmp);
294 return true;
295 }
296 }
297 {
298 float tmp;
299 if(get_value<float>(tmp)) {
300 value = signed_index_t(tmp);
301 return true;
302 }
303
304 }
305 {
306 double tmp;
307 if(get_value<double>(tmp)) {
308 value = signed_index_t(tmp);
309 return true;
310 }
311
312 }
313 return false;
314 }
315
322 bool get_value(float& value) const {
323 if(get_value<float>(value)) {
324 return true;
325 }
326 {
327 double tmp;
328 if(get_value<double>(tmp)) {
329 value = float(tmp);
330 return true;
331 }
332 }
333 {
334 index_t tmp;
335 if(get_value<index_t>(tmp)) {
336 value = float(tmp);
337 return true;
338 }
339 }
340 {
341 signed_index_t tmp;
342 if(get_value<signed_index_t>(tmp)) {
343 value = float(tmp);
344 return true;
345 }
346 }
347 return false;
348 }
349
356 bool get_value(double& value) const {
357 if(get_value<double>(value)) {
358 return true;
359 }
360 {
361 float tmp;
362 if(get_value<float>(tmp)) {
363 value = double(tmp);
364 return true;
365 }
366 }
367 {
368 index_t tmp;
369 if(get_value<index_t>(tmp)) {
370 value = double(tmp);
371 return true;
372 }
373 }
374 {
375 signed_index_t tmp;
376 if(get_value<signed_index_t>(tmp)) {
377 value = double(tmp);
378 return true;
379 }
380 }
381 return false;
382 }
383
394 template <class T> bool get_value(T*& value) const {
395 value = nullptr;
396 if(is_null()) {
397 return true;
398 }
399
400 if(is_smart_pointer_type(meta_type())) {
401 // TODO: check pointer types
402 Counted* counted = value_as<Counted*>();
403 value = reinterpret_cast<T*>(counted);
404 return true;
405 }
406
407 if(!is_pointer_type(meta_type())) {
408 Logger::warn("GOM")
409 << "Invalid Any to pointer conversion: not pointer type "
410 << std::endl;
411 Logger::warn("GOM")
412 << "type = "
413 << meta_type_name(meta_type())
414 << std::endl;
415 Logger::warn("GOM")
416 << "value = "
417 << as_string()
418 << std::endl;
419 return false;
420 }
421
422 /*
423 // TODO: check pointer types ? (but I do not want to include
424 // <meta.h> I think, to be checked).
425 if(!pointer_can_be_casted_to(meta_type(), ogf_meta<T>::type())) {
426 Logger::warn("GOM")
427 << "Invalid Any to pointer conversion: invalid cast"
428 << std::endl;
429 return false;
430 }
431 */
432
433 value = (T*)(value_as<Memory::pointer>());
434 return true;
435 }
436
442 void create(MetaType* meta_type) {
443 destroy();
444 meta_type_ = meta_type;
445 if(life_cycle()->object_size() <= BUFFER_SIZE) {
446 in_buffer_ = true;
447 value_ = buffer_;
448 life_cycle()->construct(value_);
449 } else {
450 in_buffer_ = false;
451 value_ = life_cycle()->new_object();
452 }
453 }
454
461 void copy_from(Memory::pointer addr, MetaType* meta_type) {
462 if(value_ != nullptr && this->meta_type() == meta_type) {
463 life_cycle()->assign(value_, addr);
464 return;
465 }
466 if(value_ != nullptr) {
467 destroy();
468 }
469 meta_type_ = meta_type;
470 if(life_cycle()->object_size() <= BUFFER_SIZE) {
471 in_buffer_ = true;
472 value_ = buffer_;
473 life_cycle()->copy_construct(value_, addr);
474 } else {
475 in_buffer_ = false;
476 value_ = life_cycle()->new_object(addr);
477 }
478 }
479
488 bool copy_to(Memory::pointer addr, MetaType* meta_type) const {
489 if(value_ == nullptr) {
490 return false;
491 }
492 if(meta_type == this->meta_type()) {
493 life_cycle()->assign(addr, value_);
494 return true;
495 }
496 // If meta types do not match, try conversions.
497 return copy_convert_to(addr, meta_type);
498 }
499
510 static void convert_to_string(
511 MetaType* meta_type,
512 std::string& string, Memory::pointer value
513 );
514
526 MetaType* meta_type,
527 const std::string& string, Memory::pointer value
528 );
529
535 static bool is_pointer_type(const MetaType* mtype);
536
542 static bool is_smart_pointer_type(const MetaType* mtype);
543
549 static MetaType* pointed_type(const MetaType* mtype);
550
562 const MetaType* derived_pointer_type,
563 const MetaType* base_pointer_type
564 );
565
577 const std::string& typeid_name
578 );
579
584 template <class T> static MetaType* resolve_meta_type() {
585 static MetaType* mtype =
586 resolve_meta_type_by_typeid_name(typeid(T).name());
587 return mtype;
588 }
589
590 protected:
591
600 bool copy_convert_to(Memory::pointer addr, MetaType* meta_type) const;
601
609 template <class T> const T& value_as() const {
610 return *(T*)(value_);
611 }
612
616 void destroy() {
617 if(value_ == nullptr) {
618 return;
619 }
620 if(in_buffer_) {
621 life_cycle()->destroy(value_);
622 in_buffer_ = false;
623 } else {
624 life_cycle()->delete_object(value_);
625 }
626 value_ = nullptr;
627 meta_type_ = nullptr;
628 }
629
634 void copy(const Any& rhs) {
635 geo_debug_assert(&rhs != this);
636 if(rhs.is_null()) {
637 destroy();
638 return;
639 }
640 meta_type_ = rhs.meta_type();
641 if(life_cycle()->object_size() <= BUFFER_SIZE) {
642 in_buffer_ = true;
643 value_ = buffer_;
644 life_cycle()->copy_construct(value_, rhs.value_);
645 } else {
646 in_buffer_ = false;
647 value_ = life_cycle()->new_object(rhs.value_);
648 }
649 }
650
651 static std::string meta_type_name(const MetaType* mt);
652
653 private:
654 Memory::pointer value_;
655 Memory::byte buffer_[BUFFER_SIZE];
656 bool in_buffer_;
657 MetaType* meta_type_;
658 };
659
660}
661
662#ifdef GEO_COMPILER_CLANG
663#pragma GCC diagnostic pop
664#endif
665
666#endif
#define geo_debug_assert(x)
Verifies that a condition is met.
Definition assert.h:196
Base class for reference-counted objects.
Definition counted.h:71
A class that stores a variable of arbitrary type.
Definition any.h:62
Any()
Any constructor.
Definition any.h:76
bool get_value(signed_index_t &value) const
Gets the stored value (signed_index_t overload).
Definition any.h:286
bool get_value(T *&value) const
Gets the stored value (pointers overload).
Definition any.h:394
static void convert_to_string(MetaType *meta_type, std::string &string, Memory::pointer value)
Converts an object of a given type into a string.
void set_value(const T &value)
Sets the value of this Any.
Definition any.h:157
static MetaType * pointed_type(const MetaType *mtype)
Gets the deferenced type.
std::string as_string() const
Gets a string representation.
Definition any.h:128
const T & value_as() const
Gets the value as a specific type.
Definition any.h:609
void set_value(const char *value)
Sets the value of this Any (overload for string literals).
Definition any.h:202
LifeCycle * life_cycle() const
Gets the LifeCycle.
void destroy()
Deallocates the stored variable.
Definition any.h:616
void reset()
Resets this Any to the initial null value.
Definition any.h:98
bool copy_convert_to(Memory::pointer addr, MetaType *meta_type) const
Tentatively converts the value stored in this Any to a type and if successful, store it at a specifie...
bool get_value(T &value) const
Gets the stored value.
Definition any.h:212
static bool pointer_can_be_casted_to(const MetaType *derived_pointer_type, const MetaType *base_pointer_type)
Tests whether a pointer type can be casted to an object type.
MetaType * meta_type() const
Gets the MetaType of the stored value.
Definition any.h:141
void copy(const Any &rhs)
Copies another Any.
Definition any.h:634
static MetaType * resolve_meta_type_by_typeid_name(const std::string &typeid_name)
Finds a MetaType by typeid name.
static bool is_smart_pointer_type(const MetaType *mtype)
Tests whether a MetaType is a smart pointer type.
bool get_value(std::string &value) const
Gets the stored value (std::string overload).
Definition any.h:239
void copy_from(Memory::pointer addr, MetaType *meta_type)
Assigns a value from a specifed address and specified type to this Any.
Definition any.h:461
bool copy_to(Memory::pointer addr, MetaType *meta_type) const
Copies the value stored in this Any at a specified address.
Definition any.h:488
static MetaType * resolve_meta_type()
Gets the MetaType associated with a type.
Definition any.h:584
bool get_value(index_t &value) const
Gets the stored value (index_t overload).
Definition any.h:250
static bool is_pointer_type(const MetaType *mtype)
Tests whether a MetaType is a pointer type.
bool get_value(float &value) const
Gets the stored value (float overload).
Definition any.h:322
Any & operator=(const Any &rhs)
Any affectation operator.
Definition any.h:116
bool is_null() const
Tests whether this Any is null.
Definition any.h:91
void create(MetaType *meta_type)
Initializes this Any with the default value of a specified MetaType.
Definition any.h:442
~Any()
Any destructor.
Definition any.h:82
Any(const Any &rhs)
Any copy constructor.
Definition any.h:106
bool get_value(double &value) const
Gets the stored value (double overload).
Definition any.h:356
static void convert_from_string(MetaType *meta_type, const std::string &string, Memory::pointer value)
Converts a string into an object of a given type.
Manages the life cycle of an object.
Definition life_cycle.h:59
The representation of a type in the Meta repository.
Definition meta_type.h:222
unsigned char byte
Unsigned byte type.
Definition memory.h:92
byte * pointer
Pointer to unsigned byte(s)
Definition memory.h:104
geo_signed_index_t signed_index_t
The type for storing and manipulating indices differences.
Definition numeric.h:343
geo_index_t index_t
The type for storing and manipulating indices.
Definition numeric.h:329
Global Graphite namespace.
Definition common.h:76
Definitions common to all include files in the gom library.