mirror of
https://github.com/ceph/ceph
synced 2025-01-25 20:45:06 +00:00
07ac5d3e74
git-svn-id: https://ceph.svn.sf.net/svnroot/ceph@1068 29311d96-e01e-0410-9327-a35deaab8ce9
144 lines
2.8 KiB
C++
144 lines
2.8 KiB
C++
// -*- 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:
|
|
map< utime_t, multiset<Context*> > scheduled; // time -> (context ...)
|
|
hash_map< Context*, utime_t > event_times; // event -> time
|
|
|
|
// get time of the next event
|
|
Context* get_next_scheduled(utime_t& when) {
|
|
if (scheduled.empty()) return 0;
|
|
map< utime_t, multiset<Context*> >::iterator it = scheduled.begin();
|
|
when = it->first;
|
|
multiset<Context*>::iterator sit = it->second.begin();
|
|
return *sit;
|
|
}
|
|
|
|
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
|
|
for (map< utime_t, multiset<Context*> >::iterator it = scheduled.begin();
|
|
it != scheduled.end();
|
|
it++) {
|
|
for (multiset<Context*>::iterator sit = it->second.begin();
|
|
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();
|
|
|
|
};
|
|
|
|
|
|
// single global instance
|
|
extern Timer g_timer;
|
|
|
|
|
|
|
|
#endif
|