From b1b3534de041344dac42e2753fa55d8d48e9321f Mon Sep 17 00:00:00 2001
From: Jan Travnicek <Jan.Travnicek@fit.cvut.cz>
Date: Thu, 22 Jun 2017 13:12:17 +0200
Subject: [PATCH] attempt to fix strict aliasing error

---
 alib2std/src/extensions/tree_base.hpp | 139 +++++++++++++++++---------
 1 file changed, 89 insertions(+), 50 deletions(-)

diff --git a/alib2std/src/extensions/tree_base.hpp b/alib2std/src/extensions/tree_base.hpp
index be705b975c..7d17b5c5e0 100644
--- a/alib2std/src/extensions/tree_base.hpp
+++ b/alib2std/src/extensions/tree_base.hpp
@@ -72,25 +72,40 @@ public:
 
 template < class Data, int arity, class ConstData = Data, class Cast = Data >
 class AnyaryNode {
-	typename TupleBuilder < Data, arity >::type children;
+	typedef typename TupleBuilder < Data, arity >::type ChildrenType;
+	typedef typename TupleBuilder < ConstData, arity >::type ConstChildrenType;
+
+	union ChildrenUnion {
+		ChildrenType children;
+		ConstChildrenType const_children;
+
+		ChildrenUnion ( ) : children ( ) { }
+		ChildrenUnion ( ChildrenType data ) : children ( std::move ( data ) ) { }
+
+		~ChildrenUnion ( ) {
+			children.~ChildrenType();
+		}
+	};
+
+	ChildrenUnion children_union;
 
 	template < std::size_t ... Indices >
 	void setParent ( std::index_sequence < Indices ... > ) {
-		( void ) std::initializer_list < void * > { std::get < Indices > ( children )->parent = static_cast < Cast * > ( this ) ... };
+		( void ) std::initializer_list < void * > { std::get < Indices > ( children_union.children )->parent = static_cast < Cast * > ( this ) ... };
 	}
 
 public:
-	AnyaryNode ( typename TupleBuilder < Data, arity >::type c ) : children ( std::move ( c ) ) {
+	AnyaryNode ( typename TupleBuilder < Data, arity >::type c ) : children_union ( std::move ( c ) ) {
 		setParent ( std::make_index_sequence < arity > ( ) );
 	}
 
 	virtual ~AnyaryNode ( ) noexcept {
 	}
 
-	AnyaryNode ( const AnyaryNode & other ) : AnyaryNode ( other.children ) {
+	AnyaryNode ( const AnyaryNode & other ) : AnyaryNode ( other.children_union.children ) {
 	}
 
-	AnyaryNode ( AnyaryNode && other ) noexcept : AnyaryNode ( std::move ( other.children ) ) {
+	AnyaryNode ( AnyaryNode && other ) noexcept : AnyaryNode ( std::move ( other.children_union.children ) ) {
 	}
 
 	AnyaryNode & operator =( const AnyaryNode & other ) {
@@ -98,7 +113,7 @@ public:
 	}
 
 	AnyaryNode & operator =( AnyaryNode && other ) noexcept {
-		std::swap ( this->children, other.children );
+		std::swap ( this->children_union.children, other.children_union.children );
 
 		setParent ( std::make_index_sequence < arity > ( ) );
 
@@ -106,32 +121,32 @@ public:
 	}
 
 	const typename TupleBuilder < Data, arity >::type & getElements ( ) {
-		return children;
+		return children_union.children;
 	}
 
 	const typename TupleBuilder < ConstData, arity >::type & getElements ( ) const {
-		return reinterpret_cast < const typename TupleBuilder < ConstData, arity >::type & > ( children );
+		return children_union.const_children;
 	}
 
 	template < int N >
 	const ConstData & getElement ( ) const {
-		return reinterpret_cast < const ConstData & > ( std::get < N > ( children ) );
+		return std::get < N > ( children_union.const_children );
 	}
 
 	template < int N >
 	Data & getElement ( ) {
-		return std::get < N > ( children );
+		return std::get < N > ( children_union.children );
 	}
 
 	void setElements ( typename TupleBuilder < Data, arity >::type c ) {
-		children = std::move ( c );
+		children_union.children = std::move ( c );
 		setParent ( std::make_index_sequence < arity > ( ) );
 	}
 
 	template < int N >
 	void setElement ( Data d ) {
-		std::get < N > ( children ) = std::move ( d );
-		std::get < N > ( children )->parent = static_cast < Cast * > ( this );
+		std::get < N > ( children_union.children ) = std::move ( d );
+		std::get < N > ( children_union.children )->parent = static_cast < Cast * > ( this );
 	}
 
 };
@@ -242,7 +257,19 @@ public:
 
 template < class Data, class ConstData = Data, class Cast = Data >
 class FixedaryNode {
-	std::vector < Data > children;
+	union ChildrenUnion {
+		std::vector < Data > children;
+		std::vector < ConstData > const_children;
+
+		ChildrenUnion ( ) : children ( ) { }
+		ChildrenUnion ( std::vector < Data > data ) : children ( std::move ( data ) ) { }
+
+		~ChildrenUnion ( ) {
+			children.~vector();
+		}
+	};
+
+	ChildrenUnion children_union;
 
 public:
 	template < typename ... Types >
@@ -250,24 +277,24 @@ public:
 		Data args[] { std::move ( data ) ... };
 
 		for ( Data & child : args )
-			children.push_back ( std::move ( child ) );
+			children_union.children.push_back ( std::move ( child ) );
 
-		for ( Data & child : children )
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 	}
 
-	FixedaryNode ( std::vector < Data > c ) : children ( std::move ( c ) ) {
-		for ( Data & child : children )
+	FixedaryNode ( std::vector < Data > c ) : children_union ( std::move ( c ) ) {
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 	}
 
 	virtual ~FixedaryNode ( ) noexcept {
 	}
 
-	FixedaryNode ( const FixedaryNode & other ) : FixedaryNode ( other.children ) {
+	FixedaryNode ( const FixedaryNode & other ) : FixedaryNode ( other.children_union.children ) {
 	}
 
-	FixedaryNode ( FixedaryNode && other ) noexcept : FixedaryNode ( std::move ( other.children ) ) {
+	FixedaryNode ( FixedaryNode && other ) noexcept : FixedaryNode ( std::move ( other.children_union.children ) ) {
 	}
 
 	FixedaryNode & operator =( const FixedaryNode & other ) {
@@ -275,63 +302,75 @@ public:
 	}
 
 	FixedaryNode & operator =( FixedaryNode && other ) noexcept {
-		std::swap ( this->children, other.children );
+		std::swap ( this->children_union.children, other.children_union.children );
 
-		for ( Data & child : children )
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 
 		return * this;
 	}
 
 	std::vector < Data > & getChildren ( ) {
-		return children;
+		return children_union.children;
 	}
 
 	const std::vector < ConstData > & getChildren ( ) const {
-		return reinterpret_cast < const std::vector < ConstData > & > ( children );
+		return children_union.const_children;
 	}
 
 	void setChildren ( std::vector < Data > c ) {
-		if ( c.size ( ) != children.size ( ) )
+		if ( c.size ( ) != children_union.children.size ( ) )
 			throw "Arity != size";
 
-		children = std::move ( c );
+		children_union.children = std::move ( c );
 
-		for ( Data & child : children )
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 	}
 
 	void setChild ( Data d, int index ) {
-		if ( children.size ( ) >= index )
+		if ( children_union.children.size ( ) >= index )
 			throw "Index out of bounds";
 
-		children[index] = std::move ( d );
+		children_union.children[index] = std::move ( d );
 
-		children[index]->parent = static_cast < Cast * > ( this );
+		children_union.children[index]->parent = static_cast < Cast * > ( this );
 	}
 
 };
 
 template < class Data, class ConstData = Data, class Cast = Data >
 class VararyNode {
-	std::vector < Data > children;
+	union ChildrenUnion {
+		std::vector < Data > children;
+		std::vector < ConstData > const_children;
+
+		ChildrenUnion ( ) : children ( ) { }
+		ChildrenUnion ( std::vector < Data > data ) : children ( std::move ( data ) ) { }
+
+		~ChildrenUnion ( ) {
+			children.~vector();
+		}
+	};
+
+	ChildrenUnion children_union;
 
 public:
 	VararyNode ( ) {
 	}
 
-	VararyNode ( std::vector < Data > c ) : children ( std::move ( c ) ) {
-		for ( Data & child : children )
+	VararyNode ( std::vector < Data > c ) : children_union ( std::move ( c ) ) {
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 	}
 
 	virtual ~VararyNode ( ) noexcept {
 	}
 
-	VararyNode ( const VararyNode & other ) : VararyNode ( other.children ) {
+	VararyNode ( const VararyNode & other ) : VararyNode ( other.children_union.children ) {
 	}
 
-	VararyNode ( VararyNode && other ) noexcept : VararyNode ( std::move ( other.children ) ) {
+	VararyNode ( VararyNode && other ) noexcept : VararyNode ( std::move ( other.children_union.children ) ) {
 	}
 
 	VararyNode & operator =( const VararyNode & other ) {
@@ -339,33 +378,33 @@ public:
 	}
 
 	VararyNode & operator =( VararyNode && other ) noexcept {
-		std::swap ( this->children, other.children );
+		std::swap ( this->children_union.children, other.children_union.children );
 
-		for ( Data & child : children )
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 
 		return * this;
 	}
 
 	std::vector < Data > & getChildren ( ) {
-		return children;
+		return children_union.children;
 	}
 
 	const std::vector < ConstData > & getChildren ( ) const {
-		return reinterpret_cast < const std::vector < ConstData > & > ( children );
+		return children_union.const_children;
 	}
 
 	void setChildren ( std::vector < Data > c ) {
-		children = std::move ( c );
+		children_union.children = std::move ( c );
 
-		for ( Data & child : children )
+		for ( Data & child : children_union.children )
 			child->parent = static_cast < Cast * > ( this );
 	}
 
 	void setChild ( Data d, int index ) {
-		children[index] = std::move ( d );
+		children_union.children[index] = std::move ( d );
 
-		children[index]->parent = static_cast < Cast * > ( this );
+		children_union.children[index]->parent = static_cast < Cast * > ( this );
 	}
 
 	void setChild ( Data d, typename std::vector < Data >::iterator it ) {
@@ -375,7 +414,7 @@ public:
 	}
 
 	typename std::vector < Data >::iterator insert ( typename std::vector < Data >::iterator it, Data d ) {
-		it = children.insert ( it, std::move ( d ) );
+		it = children_union.children.insert ( it, std::move ( d ) );
 
 		( * std::prev ( it ) )->parent = static_cast < Cast * > ( this );
 
@@ -384,13 +423,13 @@ public:
 
 	template < class InputIterator >
 	typename std::vector < Data >::iterator insert ( typename std::vector < Data >::iterator it, InputIterator first, InputIterator last ) {
-		size_t off = it - children.begin ( );
+		size_t off = it - children_union.children.begin ( );
 		size_t size = last - first;
 
-		children.insert ( it, first, last );
-		it = children.begin ( ) + off;
+		children_union.children.insert ( it, first, last );
+		it = children_union.children.begin ( ) + off;
 
-		 // TODO on g++-4.9 use: it = children->insert( it, first, last );
+		 // TODO on g++-4.9 use: it = children_union.children.insert( it, first, last );
 		typename std::vector < Data >::iterator end = it + size;
 
 		for ( ; it != end; it++ )
@@ -400,8 +439,8 @@ public:
 	}
 
 	void pushBackChild ( Data d ) {
-		children.push_back ( std::move ( d ) );
-		children[children.size ( ) - 1]->parent = static_cast < Cast * > ( this );
+		children_union.children.push_back ( std::move ( d ) );
+		children_union.children[children_union.children.size ( ) - 1]->parent = static_cast < Cast * > ( this );
 	}
 
 };
-- 
GitLab