-
Notifications
You must be signed in to change notification settings - Fork 5.9k
/
FutureImpl.h
129 lines (101 loc) · 3.1 KB
/
FutureImpl.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
#ifndef CEPH_JOURNAL_FUTURE_IMPL_H
#define CEPH_JOURNAL_FUTURE_IMPL_H
#include "include/int_types.h"
#include "common/Mutex.h"
#include "common/RefCountedObj.h"
#include "journal/Future.h"
#include <list>
#include <boost/noncopyable.hpp>
#include <boost/intrusive_ptr.hpp>
#include "include/assert.h"
class Context;
namespace journal {
class FutureImpl;
class JournalMetadata;
typedef boost::intrusive_ptr<FutureImpl> FutureImplPtr;
class FutureImpl : public RefCountedObject, boost::noncopyable {
public:
struct FlushHandler {
virtual ~FlushHandler() {}
virtual void flush(const FutureImplPtr &future) = 0;
virtual void get() = 0;
virtual void put() = 0;
};
typedef boost::intrusive_ptr<JournalMetadata> JournalMetadataPtr;
typedef boost::intrusive_ptr<FlushHandler> FlushHandlerPtr;
FutureImpl(JournalMetadataPtr journal_metadata, uint64_t tag_tid,
uint64_t entry_tid, uint64_t commit_tid);
void init(const FutureImplPtr &prev_future);
inline uint64_t get_tag_tid() const {
return m_tag_tid;
}
inline uint64_t get_entry_tid() const {
return m_entry_tid;
}
inline uint64_t get_commit_tid() const {
return m_commit_tid;
}
void flush(Context *on_safe = NULL);
void wait(Context *on_safe);
bool is_complete() const;
int get_return_value() const;
inline bool is_flush_in_progress() const {
Mutex::Locker locker(m_lock);
return (m_flush_state == FLUSH_STATE_IN_PROGRESS);
}
inline void set_flush_in_progress() {
Mutex::Locker locker(m_lock);
m_flush_state = FLUSH_STATE_IN_PROGRESS;
}
bool attach(const FlushHandlerPtr &flush_handler);
inline void detach() {
Mutex::Locker locker(m_lock);
assert(m_flush_handler);
m_flush_handler.reset();
}
inline FlushHandlerPtr get_flush_handler() const {
Mutex::Locker locker(m_lock);
return m_flush_handler;
}
void safe(int r);
private:
friend std::ostream &operator<<(std::ostream &, const FutureImpl &);
typedef std::list<Context *> Contexts;
enum FlushState {
FLUSH_STATE_NONE,
FLUSH_STATE_REQUESTED,
FLUSH_STATE_IN_PROGRESS
};
struct C_ConsistentAck : public Context {
FutureImplPtr future;
C_ConsistentAck(FutureImpl *_future) : future(_future) {}
virtual void complete(int r) {
future->consistent(r);
future.reset();
}
virtual void finish(int r) {}
};
JournalMetadataPtr m_journal_metadata;
uint64_t m_tag_tid;
uint64_t m_entry_tid;
uint64_t m_commit_tid;
mutable Mutex m_lock;
FutureImplPtr m_prev_future;
bool m_safe;
bool m_consistent;
int m_return_value;
FlushHandlerPtr m_flush_handler;
FlushState m_flush_state;
C_ConsistentAck m_consistent_ack;
Contexts m_contexts;
void consistent(int r);
void finish_unlock();
};
void intrusive_ptr_add_ref(FutureImpl::FlushHandler *p);
void intrusive_ptr_release(FutureImpl::FlushHandler *p);
std::ostream &operator<<(std::ostream &os, const FutureImpl &future);
} // namespace journal
using journal::operator<<;
#endif // CEPH_JOURNAL_FUTURE_IMPL_H