// Copyright 2018 The gVisor Authors.
//
// 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.

#ifndef GVISOR_TEST_SYSCALLS_BASE_POLL_TEST_H_
#define GVISOR_TEST_SYSCALLS_BASE_POLL_TEST_H_

#include <signal.h>
#include <sys/syscall.h>
#include <sys/types.h>
#include <syscall.h>
#include <time.h>
#include <unistd.h>

#include <memory>

#include "gtest/gtest.h"
#include "absl/synchronization/mutex.h"
#include "absl/time/time.h"
#include "test/util/logging.h"
#include "test/util/signal_util.h"
#include "test/util/thread_util.h"

namespace gvisor {
namespace testing {

// TimerThread is a cancelable timer.
class TimerThread {
 public:
  TimerThread(absl::Time deadline, pid_t tgid, pid_t tid)
      : thread_([=] {
          mu_.Lock();
          mu_.AwaitWithDeadline(absl::Condition(&cancel_), deadline);
          if (!cancel_) {
            TEST_PCHECK(tgkill(tgid, tid, SIGALRM) == 0);
          }
          mu_.Unlock();
        }) {}

  ~TimerThread() { Cancel(); }

  void Cancel() {
    absl::MutexLock ml(&mu_);
    cancel_ = true;
  }

 private:
  mutable absl::Mutex mu_;
  bool cancel_ ABSL_GUARDED_BY(mu_) = false;

  // Must be last to ensure that the destructor for the thread is run before
  // any other member of the object is destroyed.
  ScopedThread thread_;
};

// Base test fixture for poll, select, ppoll, and pselect tests.
//
// This fixture makes use of SIGALRM. The handler is saved in SetUp() and
// restored in TearDown().
class BasePollTest : public ::testing::Test {
 protected:
  BasePollTest();
  ~BasePollTest() override;

  // Sets a timer that will send a signal to the calling thread after
  // `duration`.
  void SetTimer(absl::Duration duration);

  // Returns true if the timer has fired.
  bool TimerFired() const;

  // Stops the pending timer (if any) and clear the "fired" state.
  void ClearTimer();

 private:
  // Thread that implements the timer. If the timer is stopped, timer_ is null.
  //
  // We have to use a thread for this purpose because tests using this fixture
  // expect to be interrupted by the timer signal, but itimers/alarm(2) send
  // thread-group-directed signals, which may be handled by any thread in the
  // test process.
  std::unique_ptr<TimerThread> timer_;

  // The original SIGALRM handler, to restore in destructor.
  struct sigaction original_alarm_sa_;
};

}  // namespace testing
}  // namespace gvisor

#endif  // GVISOR_TEST_SYSCALLS_BASE_POLL_TEST_H_