/* * Copyright (c) Meta Platforms, Inc. and affiliates. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. */ #pragma once #include #include #include "cpp/HsStructDefines.h" #include "cpp/Marshallable.h" // HsOption ------------------------------------------------------------------- #define hsc_derive_hs_option_unsafe(cxx_name...) \ hsc_printf( \ "deriveHsOptionUnsafe \"%s\" %lu ", \ #cxx_name, \ (unsigned long)sizeof(HsOption)); \ hsc_printf("%lu", (unsigned long)alignof(HsOption)); #define HS_OPTION_H(Name, Type) \ namespace hs_option { \ using Name = Type; \ } #define HS_OPTION_CPP(Name, Type...) \ extern "C" void* option_newHsOption##Name(Type* v) noexcept { \ return new HsOption(std::move(*v)); \ } \ extern "C" void option_ctorHsOption##Name(void* ret, Type* v) noexcept { \ new (ret) HsOption(std::move(*v)); \ } \ HS_DEFINE_MARSHALLABLE(HsOption##Name, FB_SINGLE_ARG(HsOption)) template HS_STRUCT HsOption { T* value = nullptr; std::optional opt = std::nullopt; public: HsOption() {} /* implicit */ HsOption(T && value) : opt(std::move(value)) { update(); } template /* implicit */ HsOption(folly::Optional && value) : opt(std::move(value)) { update(); } template /* implicit */ HsOption(std::optional && value) : opt(std::move(value)) { update(); } HsOption(const HsOption&) = delete; HsOption(HsOption && other) noexcept : opt(std::move(other.opt)) { update(); other.opt = std::nullopt; // a moved optional doesn't empty the value other.update(); } HsOption& operator=(const HsOption&) = delete; HsOption& operator=(HsOption&& other) noexcept { opt = std::move(other.opt); update(); other.opt = std::nullopt; // a moved optional doesn't empty the value other.update(); return *this; } std::optional toStdOptional()&& { std::optional res = std::move(opt); opt = std::nullopt; update(); return res; } private: void update() { if (opt.has_value()) { value = &opt.value(); } else { value = nullptr; } } };