/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include #include #include namespace folly { namespace detail { // thunk // // A carefully curated collection of generic general-purpose thunk templates: // * make: operator new with given arguments // * ruin: operator delete // * make_copy: operator new with copy-ctor // * make_move: operator new with move-ctor // * ctor: in-place constructor with given arguments // * dtor: in-place destructor // * ctor_copy: in-place copy-constructor // * ctor_move: in-place move-constructor // * noop: no-op function with the given arguments // * fail: terminating function with the given return and arguments struct thunk { template static void* make(A... a) { return new T(static_cast(a)...); } template static void ruin(void* const ptr) noexcept { delete static_cast(ptr); } template static void* make_copy(void const* const src) { return new T(*reinterpret_cast(src)); } template static void* make_move(void* const src) { return new T(static_cast(*reinterpret_cast(src))); } template static void* ctor(void* const ptr, A... a) { return ::new (ptr) T(static_cast(a)...); } template static void dtor(void* const ptr) noexcept { static_cast(ptr)->~T(); } template static void* ctor_copy(void* const dst, void const* const src) noexcept( noexcept(T(FOLLY_DECLVAL(T const&)))) { return ::new (dst) T(*reinterpret_cast(src)); } template static void* ctor_move(void* const dst, void* const src) noexcept( noexcept(T(FOLLY_DECLVAL(T&&)))) { return ::new (dst) T(static_cast(*reinterpret_cast(src))); } template static void* operator_new() { return folly::operator_new(Size, std::align_val_t(Align)); } template static void* operator_new_nx() { return folly::operator_new(Size, std::align_val_t(Align), std::nothrow); } template static void operator_delete(void* const ptr) noexcept { return folly::operator_delete(ptr, Size, std::align_val_t(Align)); } template static void noop(A...) noexcept {} template static R fail(A...) noexcept { std::terminate(); } }; } // namespace detail } // namespace folly