/*
    This file is part of TON Blockchain Library.
    TON Blockchain Library is free software: you can redistribute it and/or modify
    it under the terms of the GNU Lesser General Public License as published by
    the Free Software Foundation, either version 2 of the License, or
    (at your option) any later version.
    TON Blockchain Library is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU Lesser General Public License for more details.
    You should have received a copy of the GNU Lesser General Public License
    along with TON Blockchain Library.  If not, see .
    Copyright 2017-2020 Telegram Systems LLP
*/
#pragma once
#include 
#include "td/utils/MovableValue.h"
#include 
namespace td {
template 
class SharedState {
 public:
  friend class Guard;
  class Guard {
   public:
    Guard(Guard &&) = default;
    Guard(SharedState *self) : self(self) {
      CHECK(!self->data_->is_locked.exchange(true));
    }
    ~Guard() {
      if (self.get()) {
        CHECK(self.get()->data_->is_locked.exchange(false));
      }
    }
    T *get() {
      return &self.get()->data_->data;
    }
    T *operator->() {
      return get();
    }
   private:
    td::MovableValue *> self;
  };
  auto lock() {
    return Guard{this};
  }
  auto unsafe() {
    return &data_->data;
  }
 private:
  struct Data {
    std::atomic is_locked{};
    T data;
  };
  std::shared_ptr data_{std::make_shared()};
};
}  // namespace td