-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathsort.hpp
98 lines (80 loc) · 3.41 KB
/
sort.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
// Copyright (c) 2019-2020 Daniel Frey
// Please see LICENSE for license or visit https://github.com/taocpp/sequences/
#ifndef TAO_SEQ_SORT_HPP
#define TAO_SEQ_SORT_HPP
#include <cstddef>
#include <type_traits>
#include "at_index.hpp"
#include "make_integer_sequence.hpp"
#include "sequence_helper.hpp"
namespace tao
{
namespace sequence
{
namespace impl
{
template< typename... >
struct merge;
template< typename OP, typename T, T A, T... As, T... Rs >
struct merge< OP, integer_sequence< T, A, As... >, integer_sequence< T >, integer_sequence< T, Rs... > >
{
using type = integer_sequence< T, Rs..., A, As... >;
};
template< typename OP, typename T, T A, T... As, T B, T... Bs, T... Rs >
struct merge< OP, integer_sequence< T, A, As... >, integer_sequence< T, B, Bs... >, integer_sequence< T, Rs... > >
: conditional_t< OP::template apply< T, A, B >::value,
merge< OP, integer_sequence< T, B, Bs... >, integer_sequence< T, As... >, integer_sequence< T, Rs..., A > >,
merge< OP, integer_sequence< T, A, As... >, integer_sequence< T, Bs... >, integer_sequence< T, Rs..., B > > >
{
};
template< bool, typename... >
struct multi_merge;
template< typename OP, typename T, typename S >
struct multi_merge< false, index_sequence<>, OP, T, S >
{
using type = S;
};
template< std::size_t... Is, typename OP, typename T, typename... Ss >
struct multi_merge< true, index_sequence< Is... >, OP, T, Ss... >
: multi_merge< ( ( sizeof...( Is ) % 2 ) == 0 ),
make_index_sequence< sizeof...( Is ) / 2 >,
OP,
T,
typename merge< OP, at_index_t< Is * 2, Ss... >, at_index_t< Is * 2 + 1, Ss... >, integer_sequence< T > >::type... >
{
};
template< std::size_t... Is, typename OP, typename T, typename S, typename... Ss >
struct multi_merge< false, index_sequence< Is... >, OP, T, S, Ss... >
: multi_merge< ( ( ( sizeof...( Is ) + 1 ) % 2 ) == 0 ),
make_index_sequence< ( sizeof...( Is ) + 1 ) / 2 >,
OP,
T,
S,
typename merge< OP, at_index_t< Is * 2, Ss... >, at_index_t< Is * 2 + 1, Ss... >, integer_sequence< T > >::type... >
{
};
} // namespace impl
template< typename OP, typename T, T... Ns >
struct sort
: impl::multi_merge< ( ( sizeof...( Ns ) % 2 ) == 0 ),
make_index_sequence< sizeof...( Ns ) / 2 >,
OP,
T,
integer_sequence< T, Ns >... >
{
};
template< typename OP, typename T >
struct sort< OP, T >
{
using type = integer_sequence< T >;
};
template< typename OP, typename T, T... Ns >
struct sort< OP, integer_sequence< T, Ns... > >
: sort< OP, T, Ns... >
{
};
template< typename OP, typename T, typename impl::element_type< T >::type... Ns >
using sort_t = typename sort< OP, T, Ns... >::type;
} // namespace sequence
} // namespace tao
#endif