// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CHROMECAST_BASE_THREAD_HEALTH_CHECKER_H_
#define CHROMECAST_BASE_THREAD_HEALTH_CHECKER_H_
#include <memory>
#include "base/functional/callback.h"
#include "base/memory/ref_counted.h"
#include "base/threading/thread_checker.h"
#include "base/time/time.h"
namespace base {
class OneShotTimer;
class SequencedTaskRunner;
class TaskRunner;
} // namespace base
namespace chromecast {
// A class used to periodically check the responsiveness of a thread.
//
// The class takes two task runners, a "patient", and a "doctor". The doctor
// task runner will post a "sentinel" task to the patient and verify that it
// gets run within a certain response time, determined by |timeout|.
// If the task is not run in time, then the patient fails the health checkup and
// |on_failure| is invoked.
//
// The thread health is checked periodically, with the length between one check
// and the next determined by |interval|, and the amount of time allowed for the
// sentinel task to complete determined by |timeout|.
class ThreadHealthChecker {
public:
ThreadHealthChecker(
scoped_refptr<base::TaskRunner> patient_task_runner,
scoped_refptr<base::SequencedTaskRunner> doctor_task_runner,
base::TimeDelta interval,
base::TimeDelta timeout,
base::RepeatingClosure on_failure);
ThreadHealthChecker(const ThreadHealthChecker&) = delete;
ThreadHealthChecker& operator=(const ThreadHealthChecker&) = delete;
~ThreadHealthChecker();
private:
class Internal : public base::RefCountedThreadSafe<Internal> {
public:
Internal(scoped_refptr<base::TaskRunner> patient_task_runner,
scoped_refptr<base::SequencedTaskRunner> doctor_task_runner,
base::TimeDelta interval,
base::TimeDelta timeout,
base::RepeatingClosure on_failure);
void StartHealthCheck();
void StopHealthCheck();
private:
friend class base::RefCountedThreadSafe<Internal>;
~Internal();
void ScheduleHealthCheck();
void CheckThreadHealth();
void ThreadOk();
void ThreadTimeout();
scoped_refptr<base::TaskRunner> patient_task_runner_;
scoped_refptr<base::SequencedTaskRunner> doctor_task_runner_;
base::TimeDelta interval_;
base::TimeDelta timeout_;
std::unique_ptr<base::OneShotTimer> ok_timer_;
std::unique_ptr<base::OneShotTimer> failure_timer_;
base::RepeatingClosure on_failure_;
THREAD_CHECKER(thread_checker_);
};
scoped_refptr<base::SequencedTaskRunner> doctor_task_runner_;
scoped_refptr<Internal> internal_;
};
} // namespace chromecast
#endif // CHROMECAST_BASE_THREAD_HEALTH_CHECKER_H_