mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-02-07 06:53:33 +00:00
This replaces the STL <new> header. Mainly, it defines a global operator new and operator delete, as well as the placement versions of these. The placement versions are required to not get compile errors when trying to write a placement new statement. Each of these operators is defined with many, many different variants. A glance at new.cc is recommended followed by a chaser of the Alexandrescu talk "std::allocator is to Allocation as std::vector is to Vexation". We must provide a global-namespace source-level definition of each operator and it is illegal for any of them to be marked inline, so here we are. The upshot is that we no longer need to include <new>, and our optional/ vector headers are self-contained.
142 lines
2.8 KiB
C++
142 lines
2.8 KiB
C++
// -*-mode:c++;indent-tabs-mode:nil;c-basic-offset:4;tab-width:8;coding:utf-8-*-
|
|
// vi: set et ft=cpp ts=4 sts=4 sw=4 fenc=utf-8 :vi
|
|
#ifndef COSMOPOLITAN_CTL_OPTIONAL_H_
|
|
#define COSMOPOLITAN_CTL_OPTIONAL_H_
|
|
#include "new.h"
|
|
#include <__utility/forward.h>
|
|
#include <__utility/move.h>
|
|
#include <__utility/swap.h>
|
|
|
|
namespace ctl {
|
|
|
|
template<typename T>
|
|
class optional
|
|
{
|
|
public:
|
|
using value_type = T;
|
|
|
|
~optional()
|
|
{
|
|
if (present_)
|
|
value_.~T();
|
|
}
|
|
|
|
optional() noexcept : present_(false)
|
|
{
|
|
}
|
|
|
|
optional(const T& value) : present_(true)
|
|
{
|
|
new (&value_) T(value);
|
|
}
|
|
|
|
optional(T&& value) : present_(true)
|
|
{
|
|
new (&value_) T(std::move(value));
|
|
}
|
|
|
|
optional(const optional& other) : present_(other.present_)
|
|
{
|
|
if (other.present_)
|
|
new (&value_) T(other.value_);
|
|
}
|
|
|
|
optional(optional&& other) noexcept : present_(other.present_)
|
|
{
|
|
if (other.present_)
|
|
new (&value_) T(std::move(other.value_));
|
|
}
|
|
|
|
optional& operator=(const optional& other)
|
|
{
|
|
if (this != &other) {
|
|
reset();
|
|
if (other.present_)
|
|
new (&value_) T(other.value_);
|
|
present_ = other.present_;
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
optional& operator=(optional&& other) noexcept
|
|
{
|
|
if (this != &other) {
|
|
reset();
|
|
if (other.present_)
|
|
new (&value_) T(std::move(other.value_));
|
|
present_ = other.present_;
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
T& value() &
|
|
{
|
|
if (!present_)
|
|
__builtin_trap();
|
|
return value_;
|
|
}
|
|
|
|
const T& value() const&
|
|
{
|
|
if (!present_)
|
|
__builtin_trap();
|
|
return value_;
|
|
}
|
|
|
|
T&& value() &&
|
|
{
|
|
if (!present_)
|
|
__builtin_trap();
|
|
return std::move(value_);
|
|
}
|
|
|
|
explicit operator bool() const noexcept
|
|
{
|
|
return present_;
|
|
}
|
|
|
|
bool has_value() const noexcept
|
|
{
|
|
return present_;
|
|
}
|
|
|
|
void reset() noexcept
|
|
{
|
|
if (present_) {
|
|
value_.~T();
|
|
present_ = false;
|
|
}
|
|
}
|
|
|
|
template<typename... Args>
|
|
void emplace(Args&&... args)
|
|
{
|
|
reset();
|
|
present_ = true;
|
|
new (&value_) T(std::forward<Args>(args)...);
|
|
}
|
|
|
|
void swap(optional& other) noexcept
|
|
{
|
|
if (present_ && other.present_) {
|
|
std::swap(value_, other.value_);
|
|
} else if (present_) {
|
|
other.emplace(std::move(value_));
|
|
reset();
|
|
} else if (other.present_) {
|
|
emplace(std::move(other.value_));
|
|
other.reset();
|
|
}
|
|
}
|
|
|
|
private:
|
|
union
|
|
{
|
|
T value_;
|
|
};
|
|
bool present_;
|
|
};
|
|
|
|
} // namespace ctl
|
|
|
|
#endif // COSMOPOLITAN_CTL_OPTIONAL_H_
|