// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024 // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #pragma once #include "td/utils/common.h" #include #include #include #include namespace td { class Guard { public: Guard() = default; Guard(const Guard &) = delete; Guard &operator=(const Guard &) = delete; Guard(Guard &&) = default; Guard &operator=(Guard &&) = default; virtual ~Guard() = default; virtual void dismiss() { std::abort(); } }; template class LambdaGuard final : public Guard { public: explicit LambdaGuard(const FunctionT &func) : func_(func) { } explicit LambdaGuard(FunctionT &&func) : func_(std::move(func)) { } LambdaGuard(const LambdaGuard &) = delete; LambdaGuard &operator=(const LambdaGuard &) = delete; LambdaGuard(LambdaGuard &&other) : func_(std::move(other.func_)), dismissed_(other.dismissed_) { other.dismissed_ = true; } LambdaGuard &operator=(LambdaGuard &&) = delete; void dismiss() final { dismissed_ = true; } ~LambdaGuard() final { if (!dismissed_) { func_(); } } private: FunctionT func_; bool dismissed_ = false; }; template unique_ptr create_lambda_guard(F &&f) { return make_unique>(std::forward(f)); } template std::shared_ptr create_shared_lambda_guard(F &&f) { return std::make_shared>(std::forward(f)); } enum class ScopeExit {}; template auto operator+(ScopeExit, FunctionT &&func) { return LambdaGuard>(std::forward(func)); } } // namespace td #define SCOPE_EXIT auto TD_CONCAT(SCOPE_EXIT_VAR_, __LINE__) = ::td::ScopeExit() + [&]