2007-02-01 05:43:23 +00:00
|
|
|
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
|
|
|
|
/*
|
|
|
|
* Ceph - scalable distributed file system
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004-2006 Sage Weil <sage@newdream.net>
|
|
|
|
*
|
|
|
|
* This is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License version 2.1, as published by the Free Software
|
|
|
|
* Foundation. See file COPYING.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef __TIMER_H
|
|
|
|
#define __TIMER_H
|
|
|
|
|
|
|
|
#include "include/types.h"
|
|
|
|
#include "include/Context.h"
|
|
|
|
#include "Clock.h"
|
|
|
|
|
|
|
|
#include "Mutex.h"
|
|
|
|
#include "Cond.h"
|
|
|
|
#include "Thread.h"
|
|
|
|
|
|
|
|
#include <map>
|
|
|
|
#include <set>
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
#include <ext/hash_map>
|
|
|
|
using namespace __gnu_cxx;
|
|
|
|
|
|
|
|
|
|
|
|
/*** Timer
|
|
|
|
* schedule callbacks
|
|
|
|
*/
|
|
|
|
|
|
|
|
//class Messenger;
|
|
|
|
|
|
|
|
|
|
|
|
namespace __gnu_cxx {
|
|
|
|
template<> struct hash<Context*> {
|
|
|
|
size_t operator()(const Context *p) const {
|
|
|
|
static hash<unsigned long> H;
|
|
|
|
return H((unsigned long)p);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class Timer {
|
|
|
|
private:
|
2007-02-17 23:04:24 +00:00
|
|
|
map< utime_t, set<Context*> > scheduled; // time -> (context ...)
|
2007-02-01 05:43:23 +00:00
|
|
|
hash_map< Context*, utime_t > event_times; // event -> time
|
|
|
|
|
|
|
|
// get time of the next event
|
2007-02-17 23:04:24 +00:00
|
|
|
//Context* get_next_scheduled(utime_t& when);
|
|
|
|
|
|
|
|
bool get_next_due(utime_t &when);
|
2007-02-01 05:43:23 +00:00
|
|
|
|
|
|
|
void register_timer(); // make sure i get a callback
|
|
|
|
void cancel_timer(); // make sure i get a callback
|
|
|
|
|
|
|
|
//pthread_t thread_id;
|
|
|
|
bool thread_stop;
|
|
|
|
Mutex lock;
|
|
|
|
bool timed_sleep;
|
|
|
|
bool sleeping;
|
|
|
|
Cond sleep_cond;
|
|
|
|
Cond timeout_cond;
|
|
|
|
|
|
|
|
public:
|
|
|
|
void timer_entry(); // waiter thread (that wakes us up)
|
|
|
|
|
|
|
|
class TimerThread : public Thread {
|
|
|
|
Timer *t;
|
|
|
|
public:
|
|
|
|
void *entry() {
|
|
|
|
t->timer_entry();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
TimerThread(Timer *_t) : t(_t) {}
|
|
|
|
} timer_thread;
|
|
|
|
|
|
|
|
|
|
|
|
int num_event;
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
Timer() :
|
|
|
|
thread_stop(false),
|
|
|
|
timed_sleep(false),
|
|
|
|
sleeping(false),
|
|
|
|
timer_thread(this),
|
|
|
|
num_event(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
~Timer() {
|
|
|
|
// stop.
|
|
|
|
cancel_timer();
|
|
|
|
|
|
|
|
// scheduled
|
2007-02-17 23:04:24 +00:00
|
|
|
for (map< utime_t, set<Context*> >::iterator it = scheduled.begin();
|
2007-02-01 05:43:23 +00:00
|
|
|
it != scheduled.end();
|
|
|
|
it++) {
|
2007-02-17 23:04:24 +00:00
|
|
|
for (set<Context*>::iterator sit = it->second.begin();
|
2007-02-01 05:43:23 +00:00
|
|
|
sit != it->second.end();
|
|
|
|
sit++)
|
|
|
|
delete *sit;
|
|
|
|
}
|
|
|
|
scheduled.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void init() {
|
|
|
|
register_timer();
|
|
|
|
}
|
|
|
|
void shutdown() {
|
|
|
|
cancel_timer();
|
|
|
|
}
|
|
|
|
|
|
|
|
// schedule events
|
|
|
|
void add_event_after(float seconds,
|
|
|
|
Context *callback);
|
|
|
|
void add_event_at(utime_t when,
|
|
|
|
Context *callback);
|
|
|
|
bool cancel_event(Context *callback);
|
|
|
|
|
|
|
|
// execute pending events
|
|
|
|
void execute_pending();
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2007-02-17 23:04:24 +00:00
|
|
|
/*
|
|
|
|
* SafeTimer is a wrapper around the raw Timer (or rather, g_timer, it's global
|
|
|
|
* instantiation) that protects event execution with an existing mutex. It
|
|
|
|
* provides for, among other things, reliable event cancellation on class
|
|
|
|
* destruction. The caller just needs to cancel each event (or cancel_all()),
|
|
|
|
* and then call join() to ensure any concurrently exectuting events (in other
|
|
|
|
* threads) get flushed.
|
|
|
|
*/
|
|
|
|
class SafeTimer {
|
|
|
|
Mutex& lock;
|
|
|
|
Cond cond;
|
|
|
|
map<Context*,Context*> scheduled; // actual -> wrapper
|
|
|
|
map<Context*,Context*> canceled;
|
|
|
|
|
|
|
|
class EventWrapper : public Context {
|
|
|
|
SafeTimer *timer;
|
|
|
|
Context *actual;
|
|
|
|
public:
|
|
|
|
EventWrapper(SafeTimer *st, Context *c) : timer(st),
|
|
|
|
actual(c) {}
|
|
|
|
void finish(int r);
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
SafeTimer(Mutex& l) : lock(l) { }
|
|
|
|
~SafeTimer();
|
|
|
|
|
|
|
|
void add_event_after(float seconds, Context *c);
|
|
|
|
void add_event_at(utime_t when, Context *c);
|
|
|
|
void cancel_event(Context *c);
|
|
|
|
void cancel_all();
|
|
|
|
void join();
|
|
|
|
|
|
|
|
int get_num_scheduled() { return scheduled.size(); }
|
|
|
|
int get_num_canceled() { return canceled.size(); }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2007-02-01 05:43:23 +00:00
|
|
|
// single global instance
|
|
|
|
extern Timer g_timer;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|