/*
    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 "td/utils/common.h"
#include "td/utils/port/thread_local.h"
#include 
#include 
namespace td {
template 
class ThreadLocalStorage {
 public:
  T &get() {
    return thread_local_node().value;
  }
  template 
  void for_each(F &&f) {
    int32 n = max_thread_id_.load();
    for (int32 i = 0; i < n; i++) {
      f(nodes_[i].value);
    }
  }
  template 
  void for_each(F &&f) const {
    int32 n = max_thread_id_.load();
    for (int32 i = 0; i < n; i++) {
      f(nodes_[i].value);
    }
  }
 private:
  struct Node {
    T value{};
    char padding[TD_CONCURRENCY_PAD];
  };
  static constexpr int32 MAX_THREAD_ID = 128;
  std::atomic max_thread_id_{MAX_THREAD_ID};
  std::array nodes_;
  Node &thread_local_node() {
    auto thread_id = get_thread_id();
    CHECK(0 <= thread_id && static_cast(thread_id) < nodes_.size());
    return nodes_[thread_id];
  }
};
}  // namespace td