2017-08-30 21:42:07 +02:00
|
|
|
/** @file
|
|
|
|
* @author Edouard DUPIN
|
2017-09-11 13:27:23 +02:00
|
|
|
* @copyright 2017, Edouard DUPIN, all right reserved
|
2017-08-30 21:42:07 +02:00
|
|
|
* @license MPL v2.0 (see license file)
|
|
|
|
*/
|
|
|
|
|
2017-09-09 23:52:12 +02:00
|
|
|
#pragma once
|
2017-08-30 21:42:07 +02:00
|
|
|
#include <etk/types.hpp>
|
2017-09-11 13:27:23 +02:00
|
|
|
#include <etk/String.hpp>
|
2017-10-25 22:27:15 +02:00
|
|
|
#include <etk/Exception.hpp>
|
2017-08-30 21:42:07 +02:00
|
|
|
|
2017-09-11 13:27:23 +02:00
|
|
|
// TO facilitate debug when have a problem ...
|
|
|
|
#define ETK_FUNCTION_DEBUG(...) do {} while(false)
|
|
|
|
//#define ETK_FUNCTION_DEBUG printf
|
2017-08-30 21:42:07 +02:00
|
|
|
|
|
|
|
namespace etk {
|
2017-09-11 13:27:23 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTOR, typename ETK_TYPE_FUNCTION>
|
|
|
|
class FunctionPrivateLambda;
|
2017-09-05 23:07:32 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTION>
|
2017-09-11 13:27:23 +02:00
|
|
|
class FunctionPrivateFunction;
|
2017-09-05 23:07:32 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTION>
|
|
|
|
class FunctionPrivate;
|
|
|
|
|
|
|
|
template <typename ETK_TYPE_FUNCTION_RETURN, typename... ETK_TYPE_FUNCTION_ARGS>
|
|
|
|
class FunctionPrivate<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)> {
|
|
|
|
public:
|
|
|
|
virtual ~FunctionPrivate() {
|
|
|
|
|
|
|
|
}
|
|
|
|
virtual ETK_TYPE_FUNCTION_RETURN operator()(ETK_TYPE_FUNCTION_ARGS... _args) const = 0;
|
2017-09-11 13:27:23 +02:00
|
|
|
virtual FunctionPrivate<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)>* copy() {
|
|
|
|
ETK_FUNCTION_DEBUG(" COPY NULLPTR \n");
|
2018-06-19 22:15:52 +02:00
|
|
|
return null;
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-10-21 19:05:21 +02:00
|
|
|
virtual void copyIn(char* _buffer) {
|
|
|
|
ETK_FUNCTION_DEBUG(" COPY NULLPTR \n");
|
|
|
|
return;
|
|
|
|
}
|
2017-09-05 23:07:32 +02:00
|
|
|
};
|
2017-09-11 13:27:23 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTION_FUNCTOR, typename ETK_TYPE_FUNCTION_RETURN, typename... ETK_TYPE_FUNCTION_ARGS>
|
|
|
|
class FunctionPrivateLambda<ETK_TYPE_FUNCTION_FUNCTOR, ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)>:
|
2017-09-05 23:07:32 +02:00
|
|
|
public FunctionPrivate<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)> {
|
|
|
|
private:
|
2017-09-26 15:44:54 +02:00
|
|
|
mutable ETK_TYPE_FUNCTION_FUNCTOR m_dataPointer;
|
2017-09-05 23:07:32 +02:00
|
|
|
public:
|
2017-09-11 13:27:23 +02:00
|
|
|
FunctionPrivateLambda(ETK_TYPE_FUNCTION_FUNCTOR _functor):
|
|
|
|
m_dataPointer(_functor) {
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG(" CREATE FunctionPrivateLambda \n");
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
|
|
|
/*
|
2017-09-11 13:27:23 +02:00
|
|
|
FunctionPrivateLambda(const ETK_TYPE_FUNCTION_FUNCTOR& _functor):
|
|
|
|
m_dataPointer(_functor) {
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG(" CREATE FunctionPrivateLambda \n");
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
|
|
|
*/
|
2017-09-11 13:27:23 +02:00
|
|
|
~FunctionPrivateLambda() {
|
|
|
|
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
|
|
|
ETK_TYPE_FUNCTION_RETURN operator()(ETK_TYPE_FUNCTION_ARGS... _args) const {
|
2017-09-11 13:27:23 +02:00
|
|
|
return m_dataPointer(etk::forward<ETK_TYPE_FUNCTION_ARGS>(_args)...);
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
FunctionPrivate<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)>* copy() {
|
|
|
|
ETK_FUNCTION_DEBUG(" COPY FunctionPrivateLambda \n");
|
2017-10-19 14:51:48 +02:00
|
|
|
return ETK_NEW(FunctionPrivateLambda, m_dataPointer);
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-10-21 19:05:21 +02:00
|
|
|
virtual void copyIn(char* _buffer) {
|
|
|
|
ETK_FUNCTION_DEBUG(" COPY NULLPTR \n");
|
|
|
|
new (_buffer) FunctionPrivateLambda(m_dataPointer);
|
|
|
|
return;
|
|
|
|
}
|
2017-09-05 23:07:32 +02:00
|
|
|
};
|
2017-09-11 13:27:23 +02:00
|
|
|
|
2017-09-01 22:40:04 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTION>
|
|
|
|
class Function;
|
2017-09-02 23:20:24 +02:00
|
|
|
|
2017-09-11 13:27:23 +02:00
|
|
|
extern uint32_t MM___pppppp;
|
2017-09-01 22:40:04 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTION_RETURN, typename... ETK_TYPE_FUNCTION_ARGS>
|
|
|
|
class Function<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)> {
|
2017-09-05 23:07:32 +02:00
|
|
|
private:
|
2017-10-19 14:51:48 +02:00
|
|
|
typedef FunctionPrivate<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)> FunctionPrivateTypedef;
|
|
|
|
FunctionPrivateTypedef* m_pointerPrivate;
|
2017-10-21 19:05:21 +02:00
|
|
|
bool m_local;
|
|
|
|
char m_buffer[16];
|
2017-09-11 13:27:23 +02:00
|
|
|
uint32_t m_pppppp;
|
2017-09-05 23:07:32 +02:00
|
|
|
public:
|
|
|
|
Function():
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate(null),
|
2017-10-21 19:05:21 +02:00
|
|
|
m_local(false) {
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
2017-09-11 13:27:23 +02:00
|
|
|
m_pppppp = MM___pppppp++;
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] create Function 1 \n", m_pppppp, (uint64_t)this);
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
Function(const etk::NullPtr&):
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate(null),
|
2017-10-21 19:05:21 +02:00
|
|
|
m_local(false) {
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
2017-09-11 13:27:23 +02:00
|
|
|
m_pppppp = MM___pppppp++;
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] create Function 2\n", m_pppppp, (uint64_t)this);
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
Function(const Function& _obj):
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate(null),
|
2017-10-21 19:05:21 +02:00
|
|
|
m_local(false) {
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
2017-09-11 13:27:23 +02:00
|
|
|
m_pppppp = MM___pppppp++;
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] create Function (copy constructor) ---------------------- [%d=0X%lx]\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)&_obj);
|
2017-10-21 19:05:21 +02:00
|
|
|
if (_obj.m_local == true) {
|
|
|
|
((FunctionPrivateTypedef*)_obj.m_buffer)->copyIn(m_buffer);
|
|
|
|
m_local = true;
|
2018-06-19 22:15:52 +02:00
|
|
|
} else if (_obj.m_pointerPrivate != null) {
|
2017-09-11 13:27:23 +02:00
|
|
|
m_pointerPrivate = _obj.m_pointerPrivate->copy();
|
2017-09-09 23:52:12 +02:00
|
|
|
}
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] create Function (copy constructor) ------- (done) ------- [%d=0X%lx]\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)&_obj);
|
2017-09-11 13:27:23 +02:00
|
|
|
}
|
2017-09-05 23:07:32 +02:00
|
|
|
Function(Function&& _obj):
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate(null),
|
2017-10-21 19:05:21 +02:00
|
|
|
m_local(false) {
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
2017-09-11 13:27:23 +02:00
|
|
|
m_pppppp = MM___pppppp++;
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d] create Function 2\n", m_pppppp);
|
2017-09-11 13:27:23 +02:00
|
|
|
_obj.swap(*this);
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d] create Function 2 (done)\n", m_pppppp);
|
2017-09-11 13:27:23 +02:00
|
|
|
}
|
|
|
|
template <typename ETK_TYPE_FUNCTION_FUNCTOR,
|
|
|
|
typename etk::EnableIf< !etk::IsSame<ETK_TYPE_FUNCTION_FUNCTOR,Function>::value
|
|
|
|
&& !etk::IsSame<ETK_TYPE_FUNCTION_FUNCTOR,etk::NullPtr>::value, int
|
|
|
|
>::type = 0
|
|
|
|
>
|
|
|
|
Function(ETK_TYPE_FUNCTION_FUNCTOR _functor):
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate(null),
|
2017-10-21 19:05:21 +02:00
|
|
|
m_local(false) {
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
2017-09-11 13:27:23 +02:00
|
|
|
m_pppppp = MM___pppppp++;
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] create Function 4 \n", m_pppppp, (uint64_t)this);
|
|
|
|
typedef FunctionPrivateLambda<ETK_TYPE_FUNCTION_FUNCTOR, ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)> FunctionPrivateLambdaTypedef;
|
2017-10-21 19:05:21 +02:00
|
|
|
if (sizeof(FunctionPrivateLambdaTypedef) <= sizeof(m_buffer)) {
|
|
|
|
new(m_buffer) FunctionPrivateLambdaTypedef(_functor);
|
|
|
|
m_local = true;
|
|
|
|
} else {
|
|
|
|
m_pointerPrivate = ETK_NEW(FunctionPrivateLambdaTypedef, _functor);
|
|
|
|
}
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] create Function 4 (done)\n", m_pppppp, (uint64_t)this);
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
|
|
|
~Function() {
|
2017-09-11 13:27:23 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] DELETE Function \n", m_pppppp, (uint64_t)this);
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_DELETE(FunctionPrivateTypedef, m_pointerPrivate);
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate = null;
|
2017-10-21 19:05:21 +02:00
|
|
|
if (m_local == true) {
|
|
|
|
// force the cast:
|
|
|
|
FunctionPrivateTypedef* tmp = (FunctionPrivateTypedef*)m_buffer;
|
|
|
|
tmp->~FunctionPrivate();
|
|
|
|
m_local = false;
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
|
|
|
}
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
|
|
|
ETK_TYPE_FUNCTION_RETURN operator()(ETK_TYPE_FUNCTION_ARGS... _args) const {
|
2018-06-19 22:15:52 +02:00
|
|
|
if ( m_pointerPrivate == null
|
2017-10-21 19:05:21 +02:00
|
|
|
&& m_local == false) {
|
2018-06-19 22:15:52 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] call Function (With null !!! ==> must assert ...)\n", m_pppppp, (uint64_t)this);
|
2018-06-22 22:47:41 +02:00
|
|
|
ETK_THROW_EXCEPTION(etk::exception::NullPointerError("etk::Function call empty pointer"));
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] call Function \n", m_pppppp, (uint64_t)this);
|
2017-10-21 19:05:21 +02:00
|
|
|
if (m_local == true) {
|
|
|
|
return (*((FunctionPrivateTypedef*)m_buffer))(etk::forward<ETK_TYPE_FUNCTION_ARGS>(_args)...);
|
|
|
|
}
|
2017-09-05 23:07:32 +02:00
|
|
|
return (*m_pointerPrivate)(etk::forward<ETK_TYPE_FUNCTION_ARGS>(_args)...);
|
|
|
|
}
|
2017-10-21 19:05:21 +02:00
|
|
|
Function& operator= (const Function& _obj) {
|
2017-09-11 13:27:23 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator=(set) Function [%d=0X%lx]\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)&_obj);
|
|
|
|
Function(_obj).swap(*this);
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator=(set) Function [%d=0X%lx] (done)\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)&_obj);
|
|
|
|
return *this;
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-10-21 19:05:21 +02:00
|
|
|
Function& operator= (Function&& _obj) {
|
2017-09-11 13:27:23 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator=(move) Function [%d=0X%lx]\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)&_obj);
|
|
|
|
Function(etk::move(_obj)).swap(*this);
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator=(move) Function [%d=0X%lx] (done)\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)&_obj);
|
|
|
|
return *this;
|
2017-09-05 23:07:32 +02:00
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
Function& operator= (etk::NullPtr _obj) {
|
2017-10-19 14:51:48 +02:00
|
|
|
ETK_DELETE(FunctionPrivateTypedef, m_pointerPrivate);
|
2018-06-19 22:15:52 +02:00
|
|
|
m_pointerPrivate = null;
|
2017-10-21 19:05:21 +02:00
|
|
|
if (m_local == true) {
|
|
|
|
// force the cast:
|
|
|
|
FunctionPrivateTypedef* tmp = (FunctionPrivateTypedef*)m_buffer;
|
|
|
|
tmp->~FunctionPrivate();
|
|
|
|
m_local = false;
|
|
|
|
memset(m_buffer, 0, sizeof(m_buffer));
|
|
|
|
}
|
2018-06-19 22:15:52 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator = null 0X%lx\n", m_pppppp, (uint64_t)this, (uint64_t)m_pointerPrivate);
|
2017-09-05 23:07:32 +02:00
|
|
|
return *this;
|
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
|
|
|
|
void swap(Function& _obj) {
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] swap [%d=0X%lx]\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)_obj);
|
|
|
|
etk::swap(m_pointerPrivate, _obj.m_pointerPrivate);
|
|
|
|
etk::swap(m_pppppp, _obj.m_pppppp);
|
2017-10-21 19:05:21 +02:00
|
|
|
etk::swap(m_local, _obj.m_local);
|
|
|
|
// TODO : This is dangerous ==> to check ...
|
|
|
|
for (size_t iii=0; iii<sizeof(m_buffer); ++iii) {
|
|
|
|
char tmp = m_buffer[iii];
|
|
|
|
m_buffer[iii] = _obj.m_buffer[iii];
|
|
|
|
_obj.m_buffer[iii] = tmp;
|
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] swap [%d=0X%lx] (done)\n", m_pppppp, (uint64_t)this, _obj.m_pppppp, (uint64_t)_obj);
|
|
|
|
}
|
|
|
|
template <typename ETK_TYPE_FUNCTION_FUNCTOR,
|
|
|
|
typename etk::EnableIf< !etk::IsSame<ETK_TYPE_FUNCTION_FUNCTOR,Function>::value
|
|
|
|
&& !etk::IsSame<ETK_TYPE_FUNCTION_FUNCTOR,etk::NullPtr>::value, int
|
|
|
|
>::type = 0
|
|
|
|
>
|
|
|
|
Function& operator= (ETK_TYPE_FUNCTION_FUNCTOR&& _functor){
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator = FUNCTOR \n", m_pppppp, (uint64_t)this);
|
|
|
|
Function(etk::forward<ETK_TYPE_FUNCTION_FUNCTOR>(_functor)).swap(*this);
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator = FUNCTOR (done)\n", m_pppppp, (uint64_t)this);
|
2017-09-05 23:07:32 +02:00
|
|
|
return *this;
|
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
template <typename ETK_TYPE_FUNCTION_FUNCTOR,
|
|
|
|
typename etk::EnableIf< !etk::IsSame<ETK_TYPE_FUNCTION_FUNCTOR,Function>::value
|
|
|
|
&& !etk::IsSame<ETK_TYPE_FUNCTION_FUNCTOR,etk::NullPtr>::value, int
|
|
|
|
>::type = 0
|
|
|
|
>
|
|
|
|
Function& operator= (const ETK_TYPE_FUNCTION_FUNCTOR& _functor){
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator = const FUNCTOR& \n", m_pppppp, (uint64_t)this);
|
|
|
|
Function(_functor).swap(*this);
|
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator = const FUNCTOR& (done)\n", m_pppppp, (uint64_t)this);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
operator bool() const {
|
2018-06-19 22:15:52 +02:00
|
|
|
return m_pointerPrivate != null
|
2017-10-21 19:05:21 +02:00
|
|
|
|| m_local == true ;
|
2017-09-11 13:27:23 +02:00
|
|
|
}
|
|
|
|
bool operator!= (etk::NullPtr) const {
|
2018-06-19 22:15:52 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator != null ==> 0X%lx %s\n", m_pppppp, (uint64_t)this, (uint64_t)m_pointerPrivate, (m_pointerPrivate != null)?"true":"false");
|
|
|
|
return m_pointerPrivate != null
|
2017-10-21 19:05:21 +02:00
|
|
|
|| m_local == true;
|
2017-09-11 13:27:23 +02:00
|
|
|
}
|
|
|
|
bool operator== (etk::NullPtr) const {
|
2018-06-19 22:15:52 +02:00
|
|
|
ETK_FUNCTION_DEBUG("[%d=0X%lx] operator == null ==> 0X%lx %s\n", m_pppppp, (uint64_t)this, (uint64_t)m_pointerPrivate, (m_pointerPrivate == null)?"true":"false");
|
|
|
|
return m_pointerPrivate == null
|
2017-10-21 19:05:21 +02:00
|
|
|
&& m_local == false;
|
2017-09-11 13:27:23 +02:00
|
|
|
}
|
|
|
|
etk::String toString() const {
|
|
|
|
etk::String out = "etk::Function<..(...)>(@";
|
2017-10-21 19:05:21 +02:00
|
|
|
if (m_local == true) {
|
|
|
|
out += etk::toString((uint64_t)m_buffer);
|
|
|
|
} else {
|
|
|
|
out += etk::toString((uint64_t)m_pointerPrivate);
|
|
|
|
}
|
2017-09-11 13:27:23 +02:00
|
|
|
out += ")";
|
|
|
|
return out;
|
|
|
|
}
|
2017-08-30 21:42:07 +02:00
|
|
|
};
|
2017-09-11 13:27:23 +02:00
|
|
|
//! @not_in_doc
|
|
|
|
template <typename ETK_TYPE_FUNCTION_RETURN, typename... ETK_TYPE_FUNCTION_ARGS>
|
|
|
|
etk::Stream& operator <<(etk::Stream& _os, const etk::Function<ETK_TYPE_FUNCTION_RETURN(ETK_TYPE_FUNCTION_ARGS...)>& _obj) {
|
|
|
|
_os << _obj.toString();
|
|
|
|
return _os;
|
|
|
|
}
|
2017-09-01 22:40:04 +02:00
|
|
|
}
|