2022-12-08 15:08:02 +00:00
|
|
|
#pragma once
|
2022-12-30 14:00:02 +00:00
|
|
|
#include <luna/Badge.h>
|
2022-12-08 15:08:02 +00:00
|
|
|
#include <luna/Check.h>
|
|
|
|
#include <luna/Move.h>
|
|
|
|
#include <luna/PlacementNew.h>
|
|
|
|
|
2022-12-30 14:00:02 +00:00
|
|
|
template <typename T> class Result;
|
|
|
|
|
2022-12-08 15:08:02 +00:00
|
|
|
template <typename T> class Option
|
|
|
|
{
|
|
|
|
public:
|
2022-12-17 14:14:27 +00:00
|
|
|
Option(const T& value) : m_has_value(true)
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
|
|
|
m_storage.store_reference(value);
|
|
|
|
}
|
|
|
|
|
2022-12-17 14:14:27 +00:00
|
|
|
Option(T&& value) : m_has_value(true)
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
|
|
|
m_storage.store_moved_reference(move(value));
|
|
|
|
}
|
|
|
|
|
2022-12-17 14:14:27 +00:00
|
|
|
Option(const Option<T>& other) : m_has_value(other.m_has_value)
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
|
|
|
if (m_has_value) { m_storage.store_reference(other.m_storage.fetch_reference()); }
|
|
|
|
}
|
|
|
|
|
2022-12-17 14:14:27 +00:00
|
|
|
Option(Option<T>&& other) : m_has_value(other.m_has_value)
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
|
|
|
other.m_has_value = false;
|
|
|
|
if (m_has_value) { m_storage.store_moved_reference(move(other.m_storage.fetch_reference())); }
|
|
|
|
}
|
|
|
|
|
2022-12-30 18:02:25 +00:00
|
|
|
Option<T>& operator=(const Option<T>& other)
|
|
|
|
{
|
|
|
|
if (this == &other) return *this;
|
|
|
|
|
|
|
|
if (m_has_value) m_storage.destroy();
|
|
|
|
m_has_value = other.m_has_value;
|
|
|
|
|
|
|
|
if (m_has_value) { m_storage.store_reference(other.m_storage.fetch_reference()); }
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Option<T>& operator=(Option<T>&& other)
|
|
|
|
{
|
|
|
|
if (this == &other) return *this;
|
|
|
|
|
|
|
|
if (m_has_value) m_storage.destroy();
|
|
|
|
m_has_value = other.m_has_value;
|
|
|
|
other.m_has_value = false;
|
|
|
|
|
|
|
|
if (m_has_value) { m_storage.store_moved_reference(move(other.m_storage.fetch_reference())); }
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2022-12-17 14:14:27 +00:00
|
|
|
Option() : m_has_value(false)
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool has_value() const
|
|
|
|
{
|
|
|
|
return m_has_value;
|
|
|
|
}
|
|
|
|
|
2023-01-22 11:00:52 +00:00
|
|
|
T value(SourceLocation caller = SourceLocation::current()) const
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
2023-01-22 11:00:52 +00:00
|
|
|
expect_at(has_value(), caller, "Option::value called on an empty Option");
|
2022-12-08 15:08:02 +00:00
|
|
|
return m_storage.fetch_reference();
|
|
|
|
}
|
|
|
|
|
2022-12-30 14:00:02 +00:00
|
|
|
T unchecked_value(Badge<Result<T>>) const
|
|
|
|
{
|
|
|
|
return m_storage.fetch_reference();
|
|
|
|
}
|
|
|
|
|
2023-01-22 11:00:52 +00:00
|
|
|
T release_value(SourceLocation caller = SourceLocation::current())
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
2023-01-22 11:00:52 +00:00
|
|
|
expect_at(has_value(), caller, "Option::release_value called on an empty Option");
|
2022-12-08 15:08:02 +00:00
|
|
|
m_has_value = false;
|
2022-12-16 17:13:40 +00:00
|
|
|
return move(m_storage.fetch_reference());
|
2022-12-08 15:08:02 +00:00
|
|
|
}
|
|
|
|
|
2022-12-30 14:00:02 +00:00
|
|
|
T unchecked_release_value(Badge<Result<T>>)
|
|
|
|
{
|
|
|
|
m_has_value = false;
|
|
|
|
return move(m_storage.fetch_reference());
|
|
|
|
}
|
|
|
|
|
2022-12-08 15:08:02 +00:00
|
|
|
T value_or(const T& other) const
|
|
|
|
{
|
|
|
|
if (has_value()) return m_storage.fetch_reference();
|
|
|
|
return other;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool try_set_value(T& ref) const
|
|
|
|
{
|
|
|
|
if (!has_value()) return false;
|
|
|
|
ref = m_storage.fetch_reference();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-02-25 16:09:03 +00:00
|
|
|
bool try_move_value(T& ref)
|
2023-01-11 16:06:17 +00:00
|
|
|
{
|
|
|
|
if (!has_value()) return false;
|
|
|
|
m_has_value = false;
|
|
|
|
ref = move(m_storage.fetch_reference());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-03-11 16:41:11 +00:00
|
|
|
// NOTE: This should also exist in Result.
|
|
|
|
T* operator->()
|
|
|
|
{
|
|
|
|
expect(has_value(), "Option::operator-> called on an empty Option");
|
|
|
|
return &m_storage.fetch_reference();
|
|
|
|
}
|
|
|
|
|
2023-03-12 12:57:38 +00:00
|
|
|
T& operator*()
|
|
|
|
{
|
|
|
|
expect(has_value(), "Option::operator* called on an empty Option");
|
|
|
|
return m_storage.fetch_reference();
|
|
|
|
}
|
|
|
|
|
|
|
|
T* value_ptr(SourceLocation caller = SourceLocation::current())
|
|
|
|
{
|
|
|
|
expect_at(has_value(), caller, "Option::value_ptr called on an empty Option");
|
|
|
|
return &m_storage.fetch_reference();
|
|
|
|
}
|
|
|
|
|
2022-12-08 15:08:02 +00:00
|
|
|
~Option()
|
|
|
|
{
|
|
|
|
if (has_value()) m_storage.destroy();
|
|
|
|
}
|
|
|
|
|
2022-12-17 14:14:27 +00:00
|
|
|
// For compatibility with TRY()
|
2022-12-08 15:08:02 +00:00
|
|
|
struct ErrorHandle
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
explicit ErrorHandle()
|
|
|
|
{
|
|
|
|
}
|
2023-01-07 19:53:23 +00:00
|
|
|
|
|
|
|
friend class Option<T>;
|
2022-12-08 15:08:02 +00:00
|
|
|
};
|
|
|
|
|
2022-12-17 14:14:27 +00:00
|
|
|
Option(const ErrorHandle&) : m_has_value(false)
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-01-22 11:00:52 +00:00
|
|
|
ErrorHandle release_error(SourceLocation caller = SourceLocation::current())
|
2022-12-08 15:08:02 +00:00
|
|
|
{
|
2023-01-22 11:00:52 +00:00
|
|
|
expect_at(!has_value(), caller, "Option::release_error called on a non-empty Option");
|
2022-12-21 19:22:44 +00:00
|
|
|
return ErrorHandle {};
|
2022-12-08 15:08:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct Storage
|
|
|
|
{
|
2023-02-25 16:09:03 +00:00
|
|
|
alignas(T) u8 buffer[sizeof(T)];
|
2022-12-08 15:08:02 +00:00
|
|
|
|
|
|
|
T& fetch_reference()
|
|
|
|
{
|
2023-02-25 16:09:03 +00:00
|
|
|
return *__builtin_launder(reinterpret_cast<T*>(&buffer));
|
2022-12-08 15:08:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const T& fetch_reference() const
|
|
|
|
{
|
2023-02-25 16:09:03 +00:00
|
|
|
return *__builtin_launder(reinterpret_cast<const T*>(&buffer));
|
2022-12-08 15:08:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void store_reference(const T& ref)
|
|
|
|
{
|
|
|
|
new (buffer) T(ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
void store_moved_reference(T&& ref)
|
|
|
|
{
|
2022-12-16 17:11:17 +00:00
|
|
|
new (buffer) T(move(ref));
|
2022-12-08 15:08:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void destroy()
|
|
|
|
{
|
|
|
|
fetch_reference().~T();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Storage m_storage;
|
2022-12-21 19:22:44 +00:00
|
|
|
bool m_has_value { false };
|
2022-12-30 17:46:18 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T> inline Option<T*> nonnull_or_empty_option(T* ptr)
|
|
|
|
{
|
|
|
|
if (ptr == nullptr) return {};
|
|
|
|
else
|
|
|
|
return ptr;
|
2023-01-02 12:07:29 +00:00
|
|
|
}
|