2016-08-15 07:09:58 +08:00
|
|
|
#pragma once
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
#include <ext/aligned_buffer.h>
|
2016-08-23 02:03:45 +08:00
|
|
|
#include <cstring>
|
2016-08-15 07:09:58 +08:00
|
|
|
#include <utility>
|
2017-02-24 17:15:18 +08:00
|
|
|
#include "utils/assert.hpp"
|
2016-08-15 07:09:58 +08:00
|
|
|
|
2016-11-29 11:08:08 +08:00
|
|
|
// Optional object storage. It maybe has and maybe
|
|
|
|
// dosent have objet of type T.
|
2016-08-15 07:09:58 +08:00
|
|
|
template <class T>
|
2017-02-18 18:54:37 +08:00
|
|
|
class Option {
|
|
|
|
public:
|
|
|
|
Option() { std::memset(data._M_addr(), 0, sizeof(T)); }
|
|
|
|
|
|
|
|
Option(T const &item) {
|
|
|
|
new (data._M_addr()) T(item);
|
|
|
|
initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Option(T &&item) {
|
|
|
|
new (data._M_addr()) T(std::move(item));
|
|
|
|
initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Option(const Option &other) {
|
|
|
|
if (other.initialized) {
|
|
|
|
new (data._M_addr()) T(other.get());
|
|
|
|
initialized = true;
|
|
|
|
} else {
|
|
|
|
std::memset(data._M_addr(), 0, sizeof(T));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Containers from std which have strong exception
|
|
|
|
// guarantees wont use move constructors and operators
|
|
|
|
// wihtout noexcept. "Optimized C++,2016 , Kurt
|
|
|
|
// Guntheroth, page: 142, title: Moving instances into
|
|
|
|
// std::vector"
|
|
|
|
Option(Option &&other) noexcept {
|
|
|
|
if (other.initialized) {
|
|
|
|
new (data._M_addr()) T(std::move(other.get()));
|
|
|
|
other.initialized = false;
|
|
|
|
initialized = true;
|
|
|
|
} else {
|
|
|
|
std::memset(data._M_addr(), 0, sizeof(T));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~Option() {
|
|
|
|
if (initialized) {
|
|
|
|
get().~T();
|
|
|
|
initialized = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Option &operator=(const Option &other) {
|
|
|
|
if (initialized) {
|
|
|
|
get().~T();
|
|
|
|
initialized = false;
|
|
|
|
}
|
|
|
|
if (other.initialized) {
|
|
|
|
new (data._M_addr()) T(other.get());
|
|
|
|
initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Option &operator=(Option &&other) {
|
|
|
|
if (initialized) {
|
|
|
|
get().~T();
|
|
|
|
initialized = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (other.initialized) {
|
|
|
|
new (data._M_addr()) T(std::move(other.get()));
|
|
|
|
other.initialized = false;
|
|
|
|
initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// True if object i present.
|
|
|
|
bool is_present() const { return initialized; }
|
|
|
|
|
|
|
|
T &get() noexcept {
|
2017-02-24 17:15:18 +08:00
|
|
|
debug_assert(initialized, "Not initialized.");
|
2017-02-18 18:54:37 +08:00
|
|
|
return *data._M_ptr();
|
|
|
|
}
|
|
|
|
|
|
|
|
T &get_or(T &other) {
|
|
|
|
if (is_present()) {
|
|
|
|
return get();
|
|
|
|
} else {
|
|
|
|
return other;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns ref to object if present else other.
|
|
|
|
T const &get_or(T const &other) const {
|
|
|
|
if (is_present()) {
|
|
|
|
return get();
|
|
|
|
} else {
|
|
|
|
return other;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const T &get() const noexcept {
|
2017-02-24 17:15:18 +08:00
|
|
|
debug_assert(initialized, "Not initialized.");
|
2017-02-18 18:54:37 +08:00
|
|
|
return *data._M_ptr();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class U>
|
|
|
|
Option<U> map() {
|
|
|
|
if (is_present()) {
|
|
|
|
return Option<U>(U(take()));
|
|
|
|
} else {
|
|
|
|
return Option<U>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class U, class F>
|
|
|
|
Option<U> map(F f) {
|
|
|
|
if (is_present()) {
|
|
|
|
return Option<U>(f(take()));
|
|
|
|
} else {
|
|
|
|
return Option<U>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class U, class F>
|
|
|
|
U map_or(F f, U &&def) {
|
|
|
|
if (is_present()) {
|
|
|
|
return f(take());
|
|
|
|
} else {
|
|
|
|
return std::move(def);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class U, class F>
|
|
|
|
U call_or(F f, U &&def) {
|
|
|
|
if (is_present()) {
|
|
|
|
return f(get());
|
|
|
|
} else {
|
|
|
|
return std::move(def);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
T take() {
|
2017-02-24 17:15:18 +08:00
|
|
|
debug_assert(initialized, "Not initialized.");
|
2017-02-18 18:54:37 +08:00
|
|
|
initialized = false;
|
|
|
|
return std::move(*data._M_ptr());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Takes if it exists otherwise returns given value.
|
|
|
|
T take_or(T &&value) {
|
|
|
|
if (initialized) {
|
|
|
|
initialized = false;
|
|
|
|
return std::move(*data._M_ptr());
|
|
|
|
} else {
|
|
|
|
return std::move(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
explicit operator bool() const { return initialized; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Aligned buffer is here to ensure aligment for
|
|
|
|
// data of type T. It isn't applicable to just put T
|
|
|
|
// field because the field has to be able to be
|
|
|
|
// uninitialized to fulfill the semantics of Option class.
|
|
|
|
__gnu_cxx::__aligned_buffer<T> data;
|
|
|
|
bool initialized = false;
|
2016-08-15 07:09:58 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <class T>
|
2017-02-18 18:54:37 +08:00
|
|
|
auto make_option() {
|
|
|
|
return Option<T>();
|
2016-08-15 07:09:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2017-02-18 18:54:37 +08:00
|
|
|
auto make_option(T &&data) {
|
|
|
|
return Option<T>(std::move(data));
|
2016-08-15 07:09:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2017-02-18 18:54:37 +08:00
|
|
|
auto make_option_const(const T &&data) {
|
|
|
|
return Option<const T>(std::move(data));
|
2016-08-15 07:09:58 +08:00
|
|
|
}
|
2016-09-19 06:22:36 +08:00
|
|
|
|
|
|
|
// HELPER FUNCTIONS
|
|
|
|
template <class R>
|
2017-02-18 18:54:37 +08:00
|
|
|
bool option_fill(Option<R> &o) {
|
|
|
|
return o.is_present() && o.get().fill();
|
2016-09-19 06:22:36 +08:00
|
|
|
}
|