summaryrefslogtreecommitdiff
path: root/src/include/rustic.hpp
blob: 155885658bf10e2cebeee58453a8a4f312e69b70 (plain)
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
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/*
 */
#pragma once

template<typename T>
class slice
{
    T*  m_first;
    unsigned int    m_len;
public:
    slice(::std::vector<T>& v):
        m_first(&v[0]),
        m_len(v.size())
    {}
    
    ::std::vector<T> to_vec() const {
        return ::std::vector<T>(begin(), end());
    }
    
    unsigned int size() const {
        return m_len;
    }
    T& operator[](unsigned int i) const {
        assert(i < m_len);
        return m_first[i];
    }
    
    T* begin() const { return m_first; }
    T* end() const { return m_first + m_len; }
};

template<typename T>
::std::ostream& operator<<(::std::ostream& os, slice<T> s) {
    if( s.size() > 0 )
    {
        bool is_first = true;
        for( const auto& i : s )
        {
            if(!is_first)
                os << ", ";
            is_first = false;
            os << i;
        }
    }
    return os;
}

namespace rust {

template<typename T>
class option
{
    char    m_data[ sizeof(T) ];
    bool    m_set;
public:
    option(T ent):
        m_set(true)
    {
        new (m_data) T(::std::move(ent));
    }
    option():
        m_set(false)
    {}
    ~option() {
        if( m_set ) {
            reinterpret_cast<T*>(m_data)->~T();
        }
    }
    
    bool is_none() const { return !m_set; }
    bool is_some() const { return m_set; }
    
    const T& unwrap() const {
        assert(is_some());
        return *reinterpret_cast<const T*>(m_data);
    }
    
    void if_set(::std::function<void (const T&)> f) const {
        if( m_set ) {
            return f(m_data);
        }
    }
    //template<typename U/*, class FcnSome, class FcnNone*/>
    //U match(::std::function<U(const T&)> if_some, ::Std::function<U()> if_none) const {
    //    if( m_set ) {
    //        return if_some(m_data);
    //    }
    //    else {
    //        return if_none();
    //    }
    //}
};
template<typename T>
class option<T&>
{
    T* m_ptr;
public:
    option(T& ent):
        m_ptr(&ent)
    {}
    option():
        m_ptr(nullptr)
    {}
    
    bool is_none() const { return m_ptr == nullptr; }
    bool is_some() const { return m_ptr != nullptr; }
    T& unwrap() const {
        assert(is_some());
        return *m_ptr;
    }
    void if_set(::std::function<void (const T&)> f) const {
        if( m_ptr ) {
            return f(*m_ptr);
        }
    }
};
template<typename T>
option<T> Some(T data) {
    return option<T>( ::std::move(data) );
}
template<typename T>
option<T> None() {
    return option<T>( );
}

#define IF_OPTION_SOME(bind, var, ...)    { auto __v = (var); if( var.is_some() ) { auto bind = __v.unwrap(); (void)&bind; __VA_ARGS__ } }

};