/* * 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 #include "cpp/Destructible.h" #include "cpp/HsStructDefines.h" namespace hs_std_tuple::detail { template T move_one(va_list& args) { T* l = va_arg(args, T*); return std::move(*l); } template std::tuple makeTuple(va_list& args) { return std::tuple{move_one(args)...}; } template void peek_one(Tuple* t, va_list& args) { T* l = va_arg(args, T*); if constexpr (std::is_nothrow_move_constructible_v) { // First delete the default allocated memory l->~T(); // Then in-place move initialize new (l) T(static_cast(std::move(std::get(*t)))); } else { *l = std::move(std::get(*t)); } } template void peekVals(std::tuple* t, va_list& args, std::index_sequence) { (peek_one, T, I>(t, args), ...); } } // namespace hs_std_tuple::detail #define hsc_derive_hs_std_tuple_unsafe(cxx_name...) \ hsc_printf( \ "deriveHsStdTupleUnsafe \"%s\" %lu %lu ", \ #cxx_name, \ (unsigned long)sizeof(hs_std_tuple::cxx_name), \ (unsigned long)alignof(hs_std_tuple::cxx_name)); #define HS_STD_TUPLE_H(Name, Types) \ namespace hs_std_tuple { \ using Name = HsStdTuple; \ } /** * `__VA_ARGS__` is no good when the types listed have a comma within them. * For example, `std::tuple>` breaks as * `__VA_ARGS__` would give us "int", then "HsEithergetTuplePtr(), \ args, \ std::make_index_sequence{}); \ va_end(args); \ } /** * std::tuple<> isn't default constructible, which means we can't derive * `Marshallable` for any `std::tuple`. Since we do really want default * constructibility, create this intermediate representation that introduces * default constructibility with DCHECK support when misused. */ template HS_STRUCT HsStdTuple { union U { char mem_block[sizeof(std::tuple)]; std::tuple tup; U() : mem_block() {} ~U() {} }; U data_; bool has_tuple_{false}; public: HsStdTuple() : data_(), has_tuple_(false) {} explicit HsStdTuple(std::tuple && t) { new (&data_.tup) decltype(data_.tup){std::move(t)}; has_tuple_ = true; } HsStdTuple(const HsStdTuple&) = delete; HsStdTuple& operator=(const HsStdTuple&) = delete; HsStdTuple(HsStdTuple && other) noexcept { construct(std::move(other)); } HsStdTuple& operator=(HsStdTuple&& other) noexcept { if (this != &other) { destruct(); construct(std::move(other)); } return *this; } explicit HsStdTuple(va_list & args) { new (&data_.tup) decltype(data_.tup){ hs_std_tuple::detail::makeTuple(args)}; has_tuple_ = true; } ~HsStdTuple() { destruct(); } std::tuple* getTuplePtr() { DCHECK(has_tuple_); return &data_.tup; } std::tuple toStdTuple()&& { return std::move(data_.tup); } template < typename T = HsStdTuple, typename = std::enable_if_t == 2>> auto toStdPair()&& { return std::make_pair( std::move(std::get<0>(data_.tup)), std::move(std::get<1>(data_.tup))); } static constexpr auto tuple_size = std::tuple_size_v>; private: void construct(HsStdTuple && other) noexcept { DCHECK(this != &other); has_tuple_ = other.has_tuple_; if (has_tuple_) { new (&data_.tup) decltype(data_.tup){std::move(other.data_.tup)}; } else { new (&data_.mem_block) decltype(data_.mem_block){ std::move(*other.data_.mem_block)}; } } void destruct() { if (has_tuple_) { data_.tup.~tuple(); has_tuple_ = false; } } }; namespace std { template struct tuple_size> { static constexpr size_t value = sizeof...(Ts); }; } // namespace std