#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