#include <cstddef>
#include <tuple>
#include <type_traits>
template <typename T>
struct identity { using type = T; };
template <typename T, std::size_t I>
struct indexed {};
template <std::size_t I>
using SizeT = std::integral_constant<std::size_t, I>;
template <template <typename...> class Template, typename... T>
struct lazy_apply {
using type = Template<typename T::type...>;
};
template <typename T, typename Tuple>
struct const_tuple;
template <typename Head, typename... Tail>
struct cons_tuple<Head, std::tuple<Tail...>>
: identity<std::tuple<Head, Tail...>> {};
template <typename T, typename Tuple>
using ConsTuple = typename cons_tuple<T, Tuple>::type;
template <typename T, typename U>
struct layout_first {
#ifndef _LIBCPP_VERSION
using first = T;
using second = U;
#else
using first = U;
using second = T;
#endif
constexpr static bool value = std::is_empty<first>::value
|| (!std::is_empty<second>::value && alignof(first) > alignof(second));
};
template <typename T, typename Tuple>
struct insert_sorted
: identity<std::tuple<T>> {};
template <typename T, std::size_t I, typename TFirst, typename... TSorted, std::size_t IFirst, std::size_t... ISorted>
struct insert_sorted<indexed<T, I>, std::tuple<indexed<TFirst, IFirst>, indexed<TSorted, ISorted>...>>
: std::conditional<
layout_first<T, TFirst>::value,
identity<std::tuple<indexed<T, I>, indexed<TFirst, IFirst>, indexed<TSorted, ISorted>...>>,
lazy_apply<ConsTuple, identity<indexed<TFirst, IFirst>>, insert_sorted<indexed<T, I>, std::tuple<indexed<TSorted, ISorted>...>>>
>::type {};
template <typename T, typename Tuple>
using InsertSorted = typename insert_sorted<T, Tuple>::type;
template <typename Tuple>
struct split;
template <typename... T, std::size_t... I>
struct split<std::tuple<indexed<T, I>...>> {
using types = std::tuple<T...>;
using indices = std::tuple<SizeT<I>...>;
}
template <typename T>
using ExtractTypes = typename split<T>::types;
template <typename T>
using ExtractIndices = typename split<T>::indices;
template <typename Acc, typename T>
struct perfect_layout_impl {
using tuples = ExtractTypes<Acc>;
using map = ExtractIndices<Acc>;
};
template <typename Acc, typename Head, typename... Tail>
struct perfect_layout_impl<Acc, std::tuple<Head, Tail...>>
: perfect_layout_impl<InsertSorted<Head, Acc>, std::tuple<Tail...>> {};
template <typename Acc, typename... T>
struct with_indices_impl : identity<Acc> {};
template <typename Acc, typename Head, typename... Tail>
struct with_indices_impl<std::tuple<Acc..., indexed<Head, sizeof...(Acc)>>, Tail...> {};
template <typename... T>
struct with_indices : with_indices_impl<std::tuple<>, T...> {};
template <typename... T>
using WithIndices = typename with_indices<T...>::type;
template <typename... T>
struct perfect_layout
: perfect_layout_impl<std::tuple<>, WithIndices<T...>> {};
template <typename... T>
using PerfectLayoutTuple = typename perfect_layout<T...>::tuple;
template <typename... T>
using PerfectLayoutMap = typename perfect_layout<T...>::map;
template <std::size_t I, typename T>
using TupleElement = typename std::tuple_element<I, T>::type;
template <std::size_t... Indices>
struct indices {};
template <typename T>
struct eval_indices;
template <typename... S>
struct eval_indices<std::tuple<S...>>
: identity<indices<S::value...>> {};
template <typename T>
using EvalIndices = typename eval_indices<T>::type;
template <typename... T, std::size_t... Indices>
std::tuple<TupleElement<Indices, std::tuple<T...>>...> forward_for_my_tuple(indices<Indices...>, T&&...) {
auto fwd = std::forward_as_tuple(std::forward<T>(t)...);
return std::forward_as_tuple(std::forward<TupleElement<Indices, std::tuple<T...>>>(std::get<Indices>(fwd))...);
}
template <typename... T>
struct my_tuple {
public:
my_tuple() = default;
explicit my_tuple(T const&... t)
: inner(forward_for_my_tuple(EvalIndices<PerfectLayoutMap<T...>>(), t...) {}
template <typename... U>
explicit my_tuple(U&&... u)
: inner(forward_for_my_tuple(EvalIndices<PerfectLayoutMap<T...>>(), u...) {}
template <std::size_t I, typename... T1>
friend TupleElement<I, std::tuple<T1...>>& get(my_tuple<T1...>& t) noexcept;
template <std::size_t I, typename... T1>
friend TupleElement<I, std::tuple<T1...>>&& get(my_tuple<T1...>&& t) noexcept;
template <std::size_t I, typename... T1>
friend TupleElement<I, std::tuple<T1...>> const& get(my_tuple<T1...> const& t) noexcept;
private:
PerfectLayoutTuple<T...> inner;
};
template <std::size_t I, typename... T1>
TupleElement<I, std::tuple<T1...>>& get(my_tuple<T1...>& t) noexcept {
return std::get<TupleElement<I, PerfectLayoutMap<T...>>::value>(t.inner);
}
template <std::size_t I, typename... T1>
TupleElement<I, std::tuple<T1...>>&& get(my_tuple<T1...>&& t) noexcept {
return std::get<TupleElement<I, PerfectLayoutMap<T...>>::value>(std::move(t.inner));
}
template <std::size_t I, typename... T1>
TupleElement<I, std::tuple<T1...>> const& get(my_tuple<T1...> const& t) noexcept {
return std::get<TupleElement<I, PerfectLayoutMap<T...>>::value>(t.inner);
}
/****** TESTING PART ******/
struct empty0 {};
struct empty1 {};
struct empty2 {};
template <std::size_t Size, std::size_t Align>
using Layout = typename std::aligned_storage<Size, Align>::type;
using type1 = Layout<1,1>;
using type2 = Layout<2,2>;
using type4 = Layout<4,4>;
using type8 = Layout<8,8>;
static_assert(sizeof(my_tuple<empty0, empty1, empty2, type8>) == sizeof(type8),
"Empty base class optimization should be done on prefix empties");
static_assert(sizeof(my_tuple<type8, empty0, empty1, empty2>) == sizeof(type8),
"Empty base class optimization should be done on suffix empties");
static_assert(sizeof(my_tuple<type8, empty0, empty1, empty2, type8>) == 2*sizeof(type8),
"Empty base class optimization should be done on middle empties");
static_assert(sizeof(my_tuple<empty0, empty1, empty2, type8, type4, type2, type1, type1>) == sizeof(type8) + sizeof(type4) + sizeof(type2) + 2*sizeof(type1),
"Perfect layout should not be ruined");
static_assert(sizeof(my_tuple<type1, type2, empty0, empty1, type1, type8, empty2, type4>) == sizeof(type8) + sizeof(type4) + sizeof(type2) + 2*sizeof(type1),
"Perfect layout should be arranged");
#include <cassert>
int main() {
my_tuple<int, empty0, empty1, double, empty2, int> t { 1, {}, {}, 0.5, {}, 2 };
// reordering should be invisible to the client
static_assert(std::is_same<decltype(get<0>(t)), int&>::value, "");
static_assert(std::is_same<decltype(get<1>(t)), empty0&>::value, "");
static_assert(std::is_same<decltype(get<2>(t)), empty1&>::value, "");
static_assert(std::is_same<decltype(get<3>(t)), double&>::value, "");
static_assert(std::is_same<decltype(get<4>(t)), empty2&>::value, "");
static_assert(std::is_same<decltype(get<5>(t)), int&>::value, "");
assert(get<0>(t) == 1);
assert(get<3>(t) == 0.5);
assert(get<5>(t) == 2);
get<0>(t) = { 3 };
get<3>(t) = { 2.5 };
get<5>(t) = { 5 };
assert(get<0>(t) == 3);
assert(get<3>(t) == 2.5);
assert(get<5>(t) == 4);
}
#include <cstddef>
#include <tuple>
#include <type_traits>

template <typename T>
struct identity { using type = T; };

template <typename T, std::size_t I>
struct indexed {};

template <std::size_t I>
using SizeT = std::integral_constant<std::size_t, I>;

template <template <typename...> class Template, typename... T>
struct lazy_apply {
	using type = Template<typename T::type...>;
};

template <typename T, typename Tuple>
struct const_tuple;
template <typename Head, typename... Tail>
struct cons_tuple<Head, std::tuple<Tail...>>
: identity<std::tuple<Head, Tail...>> {};
template <typename T, typename Tuple>
using ConsTuple = typename cons_tuple<T, Tuple>::type;

template <typename T, typename U>
struct layout_first {
#ifndef _LIBCPP_VERSION
	using first = T;
	using second = U;
#else
	using first = U;
	using second = T;
#endif

	constexpr static bool value = std::is_empty<first>::value
		|| (!std::is_empty<second>::value && alignof(first) > alignof(second));
};

template <typename T, typename Tuple>
struct insert_sorted
: identity<std::tuple<T>> {};
template <typename T, std::size_t I, typename TFirst, typename... TSorted, std::size_t IFirst, std::size_t... ISorted>
struct insert_sorted<indexed<T, I>, std::tuple<indexed<TFirst, IFirst>, indexed<TSorted, ISorted>...>>
: std::conditional<
	layout_first<T, TFirst>::value,
	identity<std::tuple<indexed<T, I>, indexed<TFirst, IFirst>, indexed<TSorted, ISorted>...>>,
	lazy_apply<ConsTuple, identity<indexed<TFirst, IFirst>>, insert_sorted<indexed<T, I>, std::tuple<indexed<TSorted, ISorted>...>>>
>::type {};

template <typename T, typename Tuple>
using InsertSorted = typename insert_sorted<T, Tuple>::type;

template <typename Tuple>
struct split;
template <typename... T, std::size_t... I>
struct split<std::tuple<indexed<T, I>...>> {
	using types = std::tuple<T...>;
	using indices = std::tuple<SizeT<I>...>;
}
template <typename T>
using ExtractTypes = typename split<T>::types;
template <typename T>
using ExtractIndices = typename split<T>::indices;

template <typename Acc, typename T>
struct perfect_layout_impl {
	using tuples = ExtractTypes<Acc>;
	using map = ExtractIndices<Acc>;
};

template <typename Acc, typename Head, typename... Tail>
struct perfect_layout_impl<Acc, std::tuple<Head, Tail...>>
: perfect_layout_impl<InsertSorted<Head, Acc>, std::tuple<Tail...>> {};

template <typename Acc, typename... T>
struct with_indices_impl : identity<Acc> {};
template <typename Acc, typename Head, typename... Tail>
struct with_indices_impl<std::tuple<Acc..., indexed<Head, sizeof...(Acc)>>, Tail...> {};
template <typename... T>
struct with_indices : with_indices_impl<std::tuple<>, T...> {};
template <typename... T>
using WithIndices = typename with_indices<T...>::type;

template <typename... T>
struct perfect_layout
: perfect_layout_impl<std::tuple<>, WithIndices<T...>> {};


template <typename... T>
using PerfectLayoutTuple = typename perfect_layout<T...>::tuple;
template <typename... T>
using PerfectLayoutMap = typename perfect_layout<T...>::map;

template <std::size_t I, typename T>
using TupleElement = typename std::tuple_element<I, T>::type;

template <std::size_t... Indices>
struct indices {};

template <typename T>
struct eval_indices;
template <typename... S>
struct eval_indices<std::tuple<S...>>
: identity<indices<S::value...>> {};

template <typename T>
using EvalIndices = typename eval_indices<T>::type;

template <typename... T, std::size_t... Indices>
std::tuple<TupleElement<Indices, std::tuple<T...>>...> forward_for_my_tuple(indices<Indices...>, T&&...) {
	auto fwd = std::forward_as_tuple(std::forward<T>(t)...);
	return std::forward_as_tuple(std::forward<TupleElement<Indices, std::tuple<T...>>>(std::get<Indices>(fwd))...);
}

template <typename... T>
struct my_tuple {
public:
	my_tuple() = default;
	explicit my_tuple(T const&... t)
	: inner(forward_for_my_tuple(EvalIndices<PerfectLayoutMap<T...>>(), t...) {}

	template <typename... U>
	explicit my_tuple(U&&... u)
	: inner(forward_for_my_tuple(EvalIndices<PerfectLayoutMap<T...>>(), u...) {}

	template <std::size_t I, typename... T1>
	friend TupleElement<I, std::tuple<T1...>>& get(my_tuple<T1...>& t) noexcept;
	template <std::size_t I, typename... T1>
	friend TupleElement<I, std::tuple<T1...>>&& get(my_tuple<T1...>&& t) noexcept;
	template <std::size_t I, typename... T1>
	friend TupleElement<I, std::tuple<T1...>> const& get(my_tuple<T1...> const& t) noexcept;

private:
	PerfectLayoutTuple<T...> inner;
};

template <std::size_t I, typename... T1>
TupleElement<I, std::tuple<T1...>>& get(my_tuple<T1...>& t) noexcept {
	return std::get<TupleElement<I, PerfectLayoutMap<T...>>::value>(t.inner);
}
template <std::size_t I, typename... T1>
TupleElement<I, std::tuple<T1...>>&& get(my_tuple<T1...>&& t) noexcept {
	return std::get<TupleElement<I, PerfectLayoutMap<T...>>::value>(std::move(t.inner));
}
template <std::size_t I, typename... T1>
TupleElement<I, std::tuple<T1...>> const& get(my_tuple<T1...> const& t) noexcept {
	return std::get<TupleElement<I, PerfectLayoutMap<T...>>::value>(t.inner);
}

/****** TESTING PART ******/


struct empty0 {};
struct empty1 {};
struct empty2 {};

template <std::size_t Size, std::size_t Align>
using Layout = typename std::aligned_storage<Size, Align>::type;

using type1 = Layout<1,1>;
using type2 = Layout<2,2>;
using type4 = Layout<4,4>;
using type8 = Layout<8,8>;

static_assert(sizeof(my_tuple<empty0, empty1, empty2, type8>) == sizeof(type8),
	"Empty base class optimization should be done on prefix empties");
static_assert(sizeof(my_tuple<type8, empty0, empty1, empty2>) == sizeof(type8),
	"Empty base class optimization should be done on suffix empties");
static_assert(sizeof(my_tuple<type8, empty0, empty1, empty2, type8>) == 2*sizeof(type8),
	"Empty base class optimization should be done on middle empties");


static_assert(sizeof(my_tuple<empty0, empty1, empty2, type8, type4, type2, type1, type1>) == sizeof(type8) + sizeof(type4) + sizeof(type2) + 2*sizeof(type1),
	"Perfect layout should not be ruined");
static_assert(sizeof(my_tuple<type1, type2, empty0, empty1, type1, type8, empty2, type4>) == sizeof(type8) + sizeof(type4) + sizeof(type2) + 2*sizeof(type1),
	"Perfect layout should be arranged");

#include <cassert>

int main() {
	my_tuple<int, empty0, empty1, double, empty2, int> t { 1, {}, {}, 0.5, {}, 2 };
	// reordering should be invisible to the client
	static_assert(std::is_same<decltype(get<0>(t)), int&>::value, "");
	static_assert(std::is_same<decltype(get<1>(t)), empty0&>::value, "");
	static_assert(std::is_same<decltype(get<2>(t)), empty1&>::value, "");
	static_assert(std::is_same<decltype(get<3>(t)), double&>::value, "");
	static_assert(std::is_same<decltype(get<4>(t)), empty2&>::value, "");
	static_assert(std::is_same<decltype(get<5>(t)), int&>::value, "");
	assert(get<0>(t) == 1);
	assert(get<3>(t) == 0.5);
	assert(get<5>(t) == 2);
	get<0>(t) = { 3 };
	get<3>(t) = { 2.5 };
	get<5>(t) = { 5 };
	assert(get<0>(t) == 3);
	assert(get<3>(t) == 2.5);
	assert(get<5>(t) == 4);
}
