// Copyright (c) by respective owners including Yahoo!, Microsoft, and
// individual contributors. All rights reserved. Released under a BSD (revised)
// license as described in the file LICENSE.

#pragma once

#include <queue>

// Mutex and CV cannot be used in managed C++, tell the compiler that this is unmanaged even if included in a managed
// project.
#ifdef _M_CEE
#  pragma managed(push, off)
#  undef _M_CEE
#  include <condition_variable>
#  include <mutex>
#  define _M_CEE 001
#  pragma managed(pop)
#else
#  include <condition_variable>
#  include <mutex>
#endif

namespace VW
{
template <typename T>
class thread_safe_queue
{
public:
  thread_safe_queue(size_t max_size) : max_size(max_size) {}

  bool try_pop(T& item)
  {
    std::unique_lock<std::mutex> lock(mut);
    while (object_queue.size() == 0 && !done) { is_not_empty.wait(lock); }

    if (done && object_queue.size() == 0) { return false; }

    item = std::move(object_queue.front());
    object_queue.pop();

    is_not_full.notify_all();
    return true;
  }

  void push(T item)
  {
    std::unique_lock<std::mutex> lock(mut);
    while (object_queue.size() == max_size) { is_not_full.wait(lock); }
    object_queue.push(std::move(item));

    is_not_empty.notify_all();
  }

  void set_done()
  {
    {
      std::unique_lock<std::mutex> lock(mut);
      done = true;
    }
    is_not_empty.notify_all();
    is_not_full.notify_all();
  }

  size_t size() const
  {
    std::unique_lock<std::mutex> lock(mut);
    return object_queue.size();
  }

private:
  size_t max_size;
  std::queue<T> object_queue;
  mutable std::mutex mut;

  volatile bool done = false;

  std::condition_variable is_not_full;
  std::condition_variable is_not_empty;
};
}  // namespace VW
