51 lines
1.7 KiB
C
51 lines
1.7 KiB
C
#ifndef SUPPORT_TRACKED_VALUE_H
|
|
#define SUPPORT_TRACKED_VALUE_H
|
|
|
|
#include <cassert>
|
|
|
|
struct TrackedValue {
|
|
enum State { CONSTRUCTED, MOVED_FROM, DESTROYED };
|
|
State state;
|
|
|
|
TrackedValue() : state(State::CONSTRUCTED) {}
|
|
|
|
TrackedValue(TrackedValue const& t) : state(State::CONSTRUCTED) {
|
|
assert(t.state != State::MOVED_FROM && "copying a moved-from object");
|
|
assert(t.state != State::DESTROYED && "copying a destroyed object");
|
|
}
|
|
|
|
#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
|
|
TrackedValue(TrackedValue&& t) : state(State::CONSTRUCTED) {
|
|
assert(t.state != State::MOVED_FROM && "double moving from an object");
|
|
assert(t.state != State::DESTROYED && "moving from a destroyed object");
|
|
t.state = State::MOVED_FROM;
|
|
}
|
|
#endif
|
|
|
|
TrackedValue& operator=(TrackedValue const& t) {
|
|
assert(state != State::DESTROYED && "copy assigning into destroyed object");
|
|
assert(t.state != State::MOVED_FROM && "copying a moved-from object");
|
|
assert(t.state != State::DESTROYED && "copying a destroyed object");
|
|
state = t.state;
|
|
return *this;
|
|
}
|
|
|
|
#ifndef _LIBCPP_HAS_NO_RVALUE_REFERENCES
|
|
TrackedValue& operator=(TrackedValue&& t) {
|
|
assert(state != State::DESTROYED && "move assigning into destroyed object");
|
|
assert(t.state != State::MOVED_FROM && "double moving from an object");
|
|
assert(t.state != State::DESTROYED && "moving from a destroyed object");
|
|
state = t.state;
|
|
t.state = State::MOVED_FROM;
|
|
return *this;
|
|
}
|
|
#endif
|
|
|
|
~TrackedValue() {
|
|
assert(state != State::DESTROYED && "double-destroying an object");
|
|
state = State::DESTROYED;
|
|
}
|
|
};
|
|
|
|
#endif // SUPPORT_TRACKED_VALUE_H
|