blob: b52e140351d20abd02540c62153c1ccb20161cd0 [file] [log] [blame]
// Copyright 2011 Google Inc.
//
// 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.
//
// Author: morlovich@google.com (Maksim Orlovich)
#ifndef PAGESPEED_KERNEL_THREAD_QUEUED_ALARM_H_
#define PAGESPEED_KERNEL_THREAD_QUEUED_ALARM_H_
#include "pagespeed/kernel/base/basictypes.h"
#include "pagespeed/kernel/base/function.h"
#include "pagespeed/kernel/base/scoped_ptr.h"
#include "pagespeed/kernel/thread/queued_worker_pool.h"
#include "pagespeed/kernel/thread/scheduler.h"
namespace net_instaweb {
class AbstractMutex;
// A helper for managing alarms that need to both run in a sequence and be
// cancellable (in the CancelAlarm sense) safely; note that
// QueuedWorkerPool::Sequence::AddFunction does not provide alarm awareness.
class QueuedAlarm : public Function {
public:
// Schedules a function to run at a given time in a given sequence.
// (Note that the function's invocation may be delayed by other work
// present in the sequence at time of alarm going off).
//
// This constructor must be invoked from that sequence as well.
//
// The object will be destroyed automatically when either the callback
// is invoked or the cancellation is complete. You should not free the
// sequence until one of these points is reached.
QueuedAlarm(Scheduler* scheduler,
QueuedWorkerPool::Sequence* sequence,
int64 wakeup_time_us,
Function* callback);
// Cancels the alarm. This method must be run from the sequence given to the
// constructor; and should not be called when the callback has already been
// invoked. It is suggested that as both invocations of CancelAlarm and
// the callback are deallocation points that you defensively clear any
// pointers to the QueuedAlarm object when they occur.
//
// The function's Cancel method will be invoked; but no guarantee is made
// as to when or in what thread context. The class does guarantee, however,
// that it will not access the sequence_ once CancelAlarm() completes.
void CancelAlarm();
private:
virtual ~QueuedAlarm();
// Runs in an arbitrary thread.
virtual void Run();
// Runs in the sequence case.
void SequencePortionOfRun();
// This can get invoked if our client freed the sequence upon calling
// CancelAlarm, in the case where what normally would be SequencePortionOfRun
// is already on the queue.
void SequencePortionOfRunCancelled();
scoped_ptr<AbstractMutex> mutex_;
Scheduler* scheduler_;
QueuedWorkerPool::Sequence* sequence_;
Function* callback_;
Scheduler::Alarm* alarm_;
bool canceled_;
bool queued_sequence_portion_;
};
} // namespace net_instaweb
#endif // PAGESPEED_KERNEL_THREAD_QUEUED_ALARM_H_