• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 
2 // Copyright (C) 2008-2018 Lorenzo Caminiti
3 // Distributed under the Boost Software License, Version 1.0 (see accompanying
4 // file LICENSE_1_0.txt or a copy at http://www.boost.org/LICENSE_1_0.txt).
5 // See: http://www.boost.org/doc/libs/release/libs/contract/doc/html/index.html
6 
7 #include <boost/contract.hpp>
8 #include <vector>
9 #include <utility>
10 #include <cassert>
11 
12 //[move
13 class circular_buffer :
14         private boost::contract::constructor_precondition<circular_buffer> {
15 public:
invariant() const16     void invariant() const {
17         if(!moved()) { // Do not check (some) invariants for moved-from objects.
18             BOOST_CONTRACT_ASSERT(index() < size());
19         }
20         // More invariants here that hold also for moved-from objects (e.g.,
21         // all assertions necessary to successfully destroy moved-from objects).
22     }
23 
24     // Move constructor.
circular_buffer(circular_buffer && other)25     circular_buffer(circular_buffer&& other) :
26         boost::contract::constructor_precondition<circular_buffer>([&] {
27             BOOST_CONTRACT_ASSERT(!other.moved());
28         })
29     {
30         boost::contract::check c = boost::contract::constructor(this)
__anon6a58ea2f0202null31             .postcondition([&] {
32                 BOOST_CONTRACT_ASSERT(!moved());
33                 BOOST_CONTRACT_ASSERT(other.moved());
34             })
35         ;
36 
37         move(std::forward<circular_buffer>(other));
38     }
39 
40     // Move assignment.
operator =(circular_buffer && other)41     circular_buffer& operator=(circular_buffer&& other) {
42         // Moved-from can be (move) assigned (so no pre `!moved()` here).
43         boost::contract::check c = boost::contract::public_function(this)
44             .precondition([&] {
45                 BOOST_CONTRACT_ASSERT(!other.moved());
46             })
47             .postcondition([&] {
48                 BOOST_CONTRACT_ASSERT(!moved());
49                 BOOST_CONTRACT_ASSERT(other.moved());
50             })
51         ;
52 
53         return move(std::forward<circular_buffer>(other));
54     }
55 
~circular_buffer()56     ~circular_buffer() {
57         // Moved-from can always be destroyed (in fact no preconditions).
58         boost::contract::check c = boost::contract::destructor(this);
59     }
60 
moved() const61     bool moved() const {
62         boost::contract::check c = boost::contract::public_function(this);
63         return moved_;
64     }
65 
66 private:
67     bool moved_;
68 
69     /* ... */
70 //]
71 
72 public:
circular_buffer(std::vector<char> const & data,unsigned start=0)73     explicit circular_buffer(std::vector<char> const& data,
74             unsigned start = 0) :
75         boost::contract::constructor_precondition<circular_buffer>([&] {
76             BOOST_CONTRACT_ASSERT(start < data.size());
77         }),
78         moved_(false),
79         data_(data),
80         index_(start)
81     {
82         boost::contract::check c = boost::contract::constructor(this)
__anon6a58ea2f0602null83             .postcondition([&] {
84                 BOOST_CONTRACT_ASSERT(!moved());
85             })
86         ;
87     }
88 
89     // Copy constructor.
circular_buffer(circular_buffer const & other)90     circular_buffer(circular_buffer const& other) :
91         boost::contract::constructor_precondition<circular_buffer>([&] {
92             BOOST_CONTRACT_ASSERT(!other.moved());
93         })
94     {
95         boost::contract::check c = boost::contract::constructor(this)
__anon6a58ea2f0802null96             .postcondition([&] {
97                 BOOST_CONTRACT_ASSERT(!moved());
98             })
99         ;
100 
101         copy(other);
102     }
103 
104     // Copy assignment.
operator =(circular_buffer const & other)105     circular_buffer& operator=(circular_buffer const& other) {
106         // Moved-from can be (copy) assigned (so no pre `!moved()` here).
107         boost::contract::check c = boost::contract::public_function(this)
108             .precondition([&] {
109                 BOOST_CONTRACT_ASSERT(!other.moved());
110             })
111             .postcondition([&] {
112                 BOOST_CONTRACT_ASSERT(!moved());
113             })
114         ;
115 
116         return copy(other);
117     }
118 
read()119     char read() {
120         boost::contract::check c = boost::contract::public_function(this)
121             .precondition([&] {
122                 BOOST_CONTRACT_ASSERT(!moved());
123             })
124         ;
125 
126         unsigned i = index_++;
127         if(index_ == data_.size()) index_ = 0; // Circular.
128         return data_.at(i);
129     }
130 
131 private:
copy(circular_buffer const & other)132     circular_buffer& copy(circular_buffer const& other) {
133         data_ = other.data_;
134         index_ = other.index_;
135         moved_ = false;
136         return *this;
137     }
138 
move(circular_buffer && other)139     circular_buffer& move(circular_buffer&& other) {
140         data_ = std::move(other.data_);
141         index_ = std::move(other.index_);
142         moved_ = false;
143         other.moved_ = true; // Mark moved-from object.
144         return *this;
145     }
146 
147     std::vector<char> data_;
148     unsigned index_;
149 
150 public:
index() const151     unsigned index() const {
152         boost::contract::check c = boost::contract::public_function(this);
153         return index_;
154     }
155 
size() const156     unsigned size() const {
157         boost::contract::check c = boost::contract::public_function(this);
158         return data_.size();
159     }
160 };
161 
main()162 int main() {
163     struct err {};
164     boost::contract::set_precondition_failure(
165             [] (boost::contract::from) { throw err(); });
166 
167     {
168         circular_buffer x({'a', 'b', 'c', 'd'}, 2);
169         assert(x.read() == 'c');
170 
171         circular_buffer y1 = x; // Copy constructor.
172         assert(y1.read() == 'd');
173         assert(x.read() == 'd');
174 
175         circular_buffer y2({'h'});
176         y2 = x; // Copy assignment.
177         assert(y2.read() == 'a');
178         assert(x.read() == 'a');
179 
180         circular_buffer z1 = std::move(x); // Move constructor.
181         assert(z1.read() == 'b');
182         // Calling `x.read()` would fail `!moved()` precondition.
183 
184         x = y1; // Moved-from `x` can be copy assigned.
185         assert(x.read() == 'a');
186         assert(y1.read() == 'a');
187 
188         circular_buffer z2({'k'});
189         z2 = std::move(x); // Move assignment.
190         assert(z2.read() == 'b');
191         // Calling `x.read()` would fail `!moved()` precondition.
192 
193         x = std::move(y2); // Moved-from `x` can be move assigned.
194         assert(x.read() == 'b');
195         // Calling `y2.read()` would fail `!moved()` precondition.
196 
197     } // Moved-from `y2` can be destroyed.
198 
199     return 0;
200 }
201 
202