mirror of
https://github.com/ceph/ceph
synced 2025-01-21 18:45:23 +00:00
07ac5d3e74
git-svn-id: https://ceph.svn.sf.net/svnroot/ceph@1068 29311d96-e01e-0410-9327-a35deaab8ce9
682 lines
19 KiB
C++
682 lines
19 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 __EBOFS_BUFFERCACHE_H
|
|
#define __EBOFS_BUFFERCACHE_H
|
|
|
|
#include "include/lru.h"
|
|
#include "include/Context.h"
|
|
|
|
#include "common/Clock.h"
|
|
|
|
#include "types.h"
|
|
#include "BlockDevice.h"
|
|
|
|
#include "include/interval_set.h"
|
|
|
|
class ObjectCache;
|
|
class BufferCache;
|
|
class Onode;
|
|
|
|
class BufferHead : public LRUObject {
|
|
public:
|
|
/*
|
|
* - buffer_heads should always break across disk extent boundaries
|
|
* - partial buffer_heads are always 1 block.
|
|
*/
|
|
const static int STATE_MISSING = 0; // missing; data is on disk, but not loaded.
|
|
const static int STATE_CLEAN = 1; // Rw clean
|
|
const static int STATE_DIRTY = 2; // RW dirty
|
|
const static int STATE_TX = 3; // Rw flushing to disk
|
|
const static int STATE_RX = 4; // w reading from disk
|
|
const static int STATE_PARTIAL = 5; // reading from disk, + partial content map. always 1 block.
|
|
|
|
public:
|
|
ObjectCache *oc;
|
|
|
|
bufferlist data;
|
|
|
|
ioh_t rx_ioh; //
|
|
Extent rx_from;
|
|
ioh_t tx_ioh; //
|
|
block_t tx_block;
|
|
block_t partial_tx_to;
|
|
version_t partial_tx_epoch;
|
|
|
|
map<off_t, bufferlist> partial; // partial dirty content overlayed onto incoming data
|
|
|
|
map< block_t, list<Context*> > waitfor_read;
|
|
|
|
set<BufferHead*> shadows; // shadow bh's that clone()ed me.
|
|
BufferHead* shadow_of;
|
|
|
|
private:
|
|
int ref;
|
|
int state;
|
|
|
|
public:
|
|
version_t epoch_modified;
|
|
|
|
version_t version; // current version in cache
|
|
version_t last_flushed; // last version flushed to disk
|
|
|
|
Extent object_loc; // block position _in_object_
|
|
|
|
utime_t dirty_stamp;
|
|
|
|
public:
|
|
BufferHead(ObjectCache *o) :
|
|
oc(o), //cancellable_ioh(0), tx_epoch(0),
|
|
rx_ioh(0), tx_ioh(0), tx_block(0), partial_tx_to(0), partial_tx_epoch(0),
|
|
shadow_of(0),
|
|
ref(0), state(STATE_MISSING), epoch_modified(0), version(0), last_flushed(0)
|
|
{}
|
|
~BufferHead() {
|
|
unpin_shadows();
|
|
}
|
|
|
|
ObjectCache *get_oc() { return oc; }
|
|
|
|
int get() {
|
|
assert(ref >= 0);
|
|
if (ref == 0) lru_pin();
|
|
return ++ref;
|
|
}
|
|
int put() {
|
|
assert(ref > 0);
|
|
if (ref == 1) lru_unpin();
|
|
--ref;
|
|
return ref;
|
|
}
|
|
|
|
block_t start() { return object_loc.start; }
|
|
void set_start(block_t s) { object_loc.start = s; }
|
|
block_t length() { return object_loc.length; }
|
|
void set_length(block_t l) { object_loc.length = l; }
|
|
block_t end() { return start() + length(); }
|
|
block_t last() { return end()-1; }
|
|
|
|
version_t get_version() { return version; }
|
|
void set_version(version_t v) { version = v; }
|
|
version_t get_last_flushed() { return last_flushed; }
|
|
void set_last_flushed(version_t v) {
|
|
if (v <= last_flushed) cout << "last_flushed begin set to " << v << ", was " << last_flushed << endl;
|
|
assert(v > last_flushed);
|
|
last_flushed = v;
|
|
}
|
|
|
|
utime_t get_dirty_stamp() { return dirty_stamp; }
|
|
void set_dirty_stamp(utime_t t) { dirty_stamp = t; }
|
|
|
|
void set_state(int s) {
|
|
if (s == STATE_PARTIAL || s == STATE_RX || s == STATE_TX) get();
|
|
if (state == STATE_PARTIAL || state == STATE_RX || state == STATE_TX) put();
|
|
|
|
if ((state == STATE_TX && s != STATE_TX) ||
|
|
(state == STATE_PARTIAL && s != STATE_PARTIAL))
|
|
unpin_shadows();
|
|
|
|
state = s;
|
|
}
|
|
int get_state() { return state; }
|
|
|
|
bool is_missing() { return state == STATE_MISSING; }
|
|
bool is_dirty() { return state == STATE_DIRTY; }
|
|
bool is_clean() { return state == STATE_CLEAN; }
|
|
bool is_tx() { return state == STATE_TX; }
|
|
bool is_rx() { return state == STATE_RX; }
|
|
bool is_partial() { return state == STATE_PARTIAL; }
|
|
|
|
//bool is_partial_writes() { return !partial_write.empty(); }
|
|
//void finish_partials();
|
|
//void cancel_partials();
|
|
//void queue_partial_write(block_t b);
|
|
|
|
void add_shadow(BufferHead *dup) {
|
|
shadows.insert(dup);
|
|
dup->shadow_of = this;
|
|
dup->get();
|
|
}
|
|
void remove_shadow(BufferHead *dup) {
|
|
shadows.erase(dup);
|
|
dup->shadow_of = 0;
|
|
dup->put();
|
|
}
|
|
void unpin_shadows() {
|
|
for (set<BufferHead*>::iterator p = shadows.begin();
|
|
p != shadows.end();
|
|
++p) {
|
|
//cout << "unpin shadow " << *p << endl;
|
|
(*p)->shadow_of = 0;
|
|
(*p)->put();
|
|
}
|
|
shadows.clear();
|
|
}
|
|
|
|
void copy_partial_substr(off_t start, off_t end, bufferlist& bl) {
|
|
map<off_t, bufferlist>::iterator i = partial.begin();
|
|
|
|
// skip first bits (fully to left)
|
|
while ((i->first + i->second.length() < start) &&
|
|
i != partial.end())
|
|
i++;
|
|
assert(i != partial.end());
|
|
assert(i->first <= start);
|
|
|
|
// first
|
|
unsigned bhoff = MAX(start, i->first) - i->first;
|
|
unsigned bhlen = MIN(end-start, i->second.length());
|
|
bl.substr_of( i->second, bhoff, bhlen );
|
|
|
|
off_t pos = i->first + i->second.length();
|
|
|
|
// have continuous to end?
|
|
for (i++; i != partial.end(); i++) {
|
|
if (pos >= end) break;
|
|
assert(pos == i->first);
|
|
|
|
pos = i->first + i->second.length();
|
|
|
|
if (pos <= end) { // this whole frag
|
|
bl.append( i->second );
|
|
} else { // partial end
|
|
unsigned bhlen = end-start-bl.length();
|
|
bufferlist frag;
|
|
frag.substr_of( i->second, 0, bhlen );
|
|
bl.claim_append(frag);
|
|
break; // done.
|
|
}
|
|
}
|
|
|
|
assert(pos >= end);
|
|
assert(bl.length() == (unsigned)(end-start));
|
|
}
|
|
|
|
bool have_partial_range(off_t start, off_t end) {
|
|
map<off_t, bufferlist>::iterator i = partial.begin();
|
|
|
|
// skip first bits (fully to left)
|
|
while ((i->first + i->second.length() < start) &&
|
|
i != partial.end())
|
|
i++;
|
|
if (i == partial.end()) return false;
|
|
|
|
// have start?
|
|
if (i->first > start) return false;
|
|
off_t pos = i->first + i->second.length();
|
|
|
|
// have continuous to end?
|
|
for (i++; i != partial.end(); i++) {
|
|
assert(pos <= i->first);
|
|
if (pos < i->first) return false;
|
|
assert(pos == i->first);
|
|
pos = i->first + i->second.length();
|
|
if (pos >= end) break; // gone far enough
|
|
}
|
|
|
|
if (pos >= end) return true;
|
|
return false;
|
|
}
|
|
|
|
bool partial_is_complete(off_t size) {
|
|
return have_partial_range( 0, MIN(size, EBOFS_BLOCK_SIZE) );
|
|
//(off_t)(start()*EBOFS_BLOCK_SIZE),
|
|
//MIN( size, (off_t)(end()*EBOFS_BLOCK_SIZE) ) );
|
|
}
|
|
void apply_partial() {
|
|
apply_partial(data, partial);
|
|
partial.clear();
|
|
}
|
|
static void apply_partial(bufferlist& bl, map<off_t, bufferlist>& pm) {
|
|
assert(bl.length() == (unsigned)EBOFS_BLOCK_SIZE);
|
|
//assert(partial_is_complete());
|
|
//cout << "apply_partial" << endl;
|
|
for (map<off_t, bufferlist>::iterator i = pm.begin();
|
|
i != pm.end();
|
|
i++) {
|
|
int pos = i->first;
|
|
//cout << " frag at opos " << i->first << " bhpos " << pos << " len " << i->second.length() << endl;
|
|
bl.copy_in(pos, i->second.length(), i->second);
|
|
}
|
|
pm.clear();
|
|
}
|
|
void add_partial(off_t off, bufferlist& p) {
|
|
unsigned len = p.length();
|
|
assert(len <= (unsigned)EBOFS_BLOCK_SIZE);
|
|
//assert(off >= (off_t)(start()*EBOFS_BLOCK_SIZE));
|
|
//assert(off + len <= (off_t)(end()*EBOFS_BLOCK_SIZE));
|
|
assert(off >= 0);
|
|
assert(off + len <= EBOFS_BLOCK_SIZE);
|
|
|
|
// trim any existing that overlaps
|
|
for (map<off_t, bufferlist>::iterator i = partial.begin();
|
|
i != partial.end();
|
|
) {
|
|
if (i->first + i->second.length() <= off) { // before
|
|
i++;
|
|
continue;
|
|
}
|
|
if (i->first >= off+len) break; // past affected area.
|
|
|
|
// overlap all?
|
|
if (off <= i->first && i->first + i->second.length() <= off+len) {
|
|
// erase it and move on.
|
|
off_t dead = i->first;
|
|
i++;
|
|
partial.erase(dead);
|
|
continue;
|
|
}
|
|
// overlap tail?
|
|
else if (i->first < off && off < i->first + i->second.length()) {
|
|
// shorten.
|
|
unsigned newlen = off - i->first;
|
|
bufferlist o;
|
|
o.claim( i->second );
|
|
i->second.substr_of(o, 0, newlen);
|
|
i++;
|
|
continue;
|
|
}
|
|
// overlap head?
|
|
else if (off < i->first && off+len < i->first + i->second.length()) {
|
|
// move.
|
|
off_t oldoff = i->first;
|
|
off_t newoff = off+len;
|
|
unsigned trim = newoff - oldoff;
|
|
partial[newoff].substr_of(i->second, trim, i->second.length()-trim);
|
|
i++; // should be at newoff!
|
|
partial.erase( oldoff );
|
|
i++;
|
|
continue;
|
|
} else
|
|
assert(0);
|
|
}
|
|
|
|
// insert
|
|
partial[off] = p;
|
|
}
|
|
|
|
|
|
};
|
|
|
|
inline ostream& operator<<(ostream& out, BufferHead& bh)
|
|
{
|
|
out << "bufferhead(" << bh.start() << "~" << bh.length();
|
|
out << " v" << bh.get_version() << "/" << bh.get_last_flushed();
|
|
if (bh.is_missing()) out << " missing";
|
|
if (bh.is_dirty()) out << " dirty";
|
|
if (bh.is_clean()) out << " clean";
|
|
if (bh.is_rx()) out << " rx";
|
|
if (bh.is_tx()) out << " tx";
|
|
if (bh.is_partial()) out << " partial";
|
|
//out << " " << bh.data.length();
|
|
out << " " << &bh;
|
|
out << ")";
|
|
return out;
|
|
}
|
|
|
|
|
|
class ObjectCache {
|
|
public:
|
|
object_t object_id;
|
|
Onode *on;
|
|
BufferCache *bc;
|
|
|
|
private:
|
|
map<block_t, BufferHead*> data;
|
|
int ref;
|
|
|
|
public:
|
|
version_t write_count;
|
|
|
|
|
|
public:
|
|
ObjectCache(object_t o, Onode *_on, BufferCache *b) :
|
|
object_id(o), on(_on), bc(b), ref(0),
|
|
write_count(0) { }
|
|
~ObjectCache() {
|
|
assert(data.empty());
|
|
assert(ref == 0);
|
|
}
|
|
|
|
int get() {
|
|
++ref;
|
|
//cout << "oc.get " << object_id << " " << ref << endl;
|
|
return ref;
|
|
}
|
|
int put() {
|
|
assert(ref > 0);
|
|
--ref;
|
|
//cout << "oc.put " << object_id << " " << ref << endl;
|
|
return ref;
|
|
}
|
|
|
|
object_t get_object_id() { return object_id; }
|
|
|
|
void add_bh(BufferHead *bh) {
|
|
// add to my map
|
|
assert(data.count(bh->start()) == 0);
|
|
|
|
if (0) { // sanity check FIXME DEBUG
|
|
//cout << "add_bh " << bh->start() << "~" << bh->length() << endl;
|
|
map<block_t,BufferHead*>::iterator p = data.lower_bound(bh->start());
|
|
if (p != data.end()) {
|
|
//cout << " after " << *p->second << endl;
|
|
//cout << " after starts at " << p->first << endl;
|
|
assert(p->first >= bh->end());
|
|
}
|
|
if (p != data.begin()) {
|
|
p--;
|
|
//cout << " before starts at " << p->second->start()
|
|
//<< " and ends at " << p->second->end() << endl;
|
|
//cout << " before " << *p->second << endl;
|
|
assert(p->second->end() <= bh->start());
|
|
}
|
|
}
|
|
|
|
data[bh->start()] = bh;
|
|
}
|
|
void remove_bh(BufferHead *bh) {
|
|
assert(data.count(bh->start()));
|
|
data.erase(bh->start());
|
|
}
|
|
bool is_empty() { return data.empty(); }
|
|
|
|
int find_tx(block_t start, block_t len,
|
|
list<BufferHead*>& tx);
|
|
|
|
int map_read(block_t start, block_t len,
|
|
map<block_t, BufferHead*>& hits, // hits
|
|
map<block_t, BufferHead*>& missing, // read these from disk
|
|
map<block_t, BufferHead*>& rx, // wait for these to finish reading from disk
|
|
map<block_t, BufferHead*>& partial); // (maybe) wait for these to read from disk
|
|
|
|
int map_write(block_t start, block_t len,
|
|
interval_set<block_t>& alloc,
|
|
map<block_t, BufferHead*>& hits,
|
|
version_t super_epoch); // can write to these.
|
|
|
|
BufferHead *split(BufferHead *bh, block_t off);
|
|
|
|
/*int scan_versions(block_t start, block_t len,
|
|
version_t& low, version_t& high);
|
|
*/
|
|
|
|
void rx_finish(ioh_t ioh, block_t start, block_t length, bufferlist& bl);
|
|
void tx_finish(ioh_t ioh, block_t start, block_t length, version_t v, version_t epoch);
|
|
|
|
void truncate(block_t blocks, version_t super_epoch);
|
|
// void tear_down();
|
|
|
|
void clone_to(Onode *other);
|
|
|
|
void dump() {
|
|
for (map<block_t,BufferHead*>::iterator i = data.begin();
|
|
i != data.end();
|
|
i++)
|
|
cout << "dump: " << i->first << ": " << *i->second << endl;
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class BufferCache {
|
|
public:
|
|
Mutex &ebofs_lock; // hack: this is a ref to global ebofs_lock
|
|
BlockDevice &dev;
|
|
|
|
set<BufferHead*> dirty_bh;
|
|
|
|
LRU lru_dirty, lru_rest;
|
|
|
|
private:
|
|
Cond stat_cond;
|
|
Cond flush_cond;
|
|
int stat_waiter;
|
|
|
|
off_t stat_clean;
|
|
off_t stat_dirty;
|
|
off_t stat_rx;
|
|
off_t stat_tx;
|
|
off_t stat_partial;
|
|
off_t stat_missing;
|
|
|
|
#define EBOFS_BC_FLUSH_BHWRITE 0
|
|
#define EBOFS_BC_FLUSH_PARTIAL 1
|
|
|
|
map<version_t, int> epoch_unflushed[2];
|
|
|
|
/* partial writes - incomplete blocks that can't be written until
|
|
* their prior content is read and overlayed with the new data.
|
|
*
|
|
* we put partial block management here because objects may be deleted
|
|
* before the read completes, but the write may have been committed in a
|
|
* prior epoch.
|
|
*
|
|
* we map: src block -> dest block -> PartialWrite
|
|
*
|
|
* really, at most there will only ever be two of these, for current+previous epochs.
|
|
*/
|
|
class PartialWrite {
|
|
public:
|
|
map<off_t, bufferlist> partial; // partial dirty content overlayed onto incoming data
|
|
version_t epoch;
|
|
};
|
|
|
|
map<block_t, map<block_t, PartialWrite> > partial_write; // queued writes w/ partial content
|
|
map<block_t, set<BufferHead*> > shadow_partials;
|
|
|
|
public:
|
|
BufferCache(BlockDevice& d, Mutex& el) :
|
|
ebofs_lock(el), dev(d),
|
|
stat_waiter(0),
|
|
stat_clean(0), stat_dirty(0), stat_rx(0), stat_tx(0), stat_partial(0), stat_missing(0)
|
|
{}
|
|
|
|
|
|
off_t get_size() {
|
|
return stat_clean+stat_dirty+stat_rx+stat_tx+stat_partial;
|
|
}
|
|
off_t get_trimmable() {
|
|
return stat_clean;
|
|
}
|
|
|
|
|
|
// bh's in cache
|
|
void add_bh(BufferHead *bh) {
|
|
bh->get_oc()->add_bh(bh);
|
|
if (bh->is_dirty()) {
|
|
lru_dirty.lru_insert_mid(bh);
|
|
dirty_bh.insert(bh);
|
|
} else
|
|
lru_rest.lru_insert_mid(bh);
|
|
stat_add(bh);
|
|
}
|
|
void touch(BufferHead *bh) {
|
|
if (bh->is_dirty()) {
|
|
lru_dirty.lru_touch(bh);
|
|
} else
|
|
lru_rest.lru_touch(bh);
|
|
}
|
|
void remove_bh(BufferHead *bh) {
|
|
bh->get_oc()->remove_bh(bh);
|
|
stat_sub(bh);
|
|
if (bh->is_dirty()) {
|
|
lru_dirty.lru_remove(bh);
|
|
dirty_bh.erase(bh);
|
|
} else
|
|
lru_rest.lru_remove(bh);
|
|
}
|
|
|
|
// stats
|
|
void stat_add(BufferHead *bh) {
|
|
switch (bh->get_state()) {
|
|
case BufferHead::STATE_MISSING: stat_missing += bh->length(); break;
|
|
case BufferHead::STATE_CLEAN: stat_clean += bh->length(); break;
|
|
case BufferHead::STATE_DIRTY: stat_dirty += bh->length(); break;
|
|
case BufferHead::STATE_TX: stat_tx += bh->length(); break;
|
|
case BufferHead::STATE_RX: stat_rx += bh->length(); break;
|
|
case BufferHead::STATE_PARTIAL: stat_partial += bh->length(); break;
|
|
}
|
|
if (stat_waiter) stat_cond.Signal();
|
|
}
|
|
void stat_sub(BufferHead *bh) {
|
|
switch (bh->get_state()) {
|
|
case BufferHead::STATE_MISSING: stat_missing -= bh->length(); break;
|
|
case BufferHead::STATE_CLEAN: stat_clean -= bh->length(); break;
|
|
case BufferHead::STATE_DIRTY: stat_dirty -= bh->length(); break;
|
|
case BufferHead::STATE_TX: stat_tx -= bh->length(); break;
|
|
case BufferHead::STATE_RX: stat_rx -= bh->length(); break;
|
|
case BufferHead::STATE_PARTIAL: stat_partial -= bh->length(); break;
|
|
}
|
|
}
|
|
off_t get_stat_tx() { return stat_tx; }
|
|
off_t get_stat_rx() { return stat_rx; }
|
|
off_t get_stat_dirty() { return stat_dirty; }
|
|
off_t get_stat_clean() { return stat_clean; }
|
|
off_t get_stat_partial() { return stat_partial; }
|
|
|
|
|
|
map<version_t, int> &get_unflushed(int what) {
|
|
return epoch_unflushed[what];
|
|
}
|
|
|
|
int get_unflushed(int what, version_t epoch) {
|
|
return epoch_unflushed[what][epoch];
|
|
}
|
|
void inc_unflushed(int what, version_t epoch) {
|
|
epoch_unflushed[what][epoch]++;
|
|
//cout << "inc_unflushed " << epoch << " now " << epoch_unflushed[epoch] << endl;
|
|
}
|
|
void dec_unflushed(int what, version_t epoch) {
|
|
epoch_unflushed[what][epoch]--;
|
|
//cout << "dec_unflushed " << epoch << " now " << epoch_unflushed[epoch] << endl;
|
|
if (epoch_unflushed[what][epoch] == 0)
|
|
flush_cond.Signal();
|
|
}
|
|
|
|
void waitfor_stat() {
|
|
stat_waiter++;
|
|
stat_cond.Wait(ebofs_lock);
|
|
stat_waiter--;
|
|
}
|
|
void waitfor_flush() {
|
|
flush_cond.Wait(ebofs_lock);
|
|
}
|
|
|
|
|
|
// bh state
|
|
void set_state(BufferHead *bh, int s) {
|
|
// move between lru lists?
|
|
if (s == BufferHead::STATE_DIRTY && bh->get_state() != BufferHead::STATE_DIRTY) {
|
|
lru_rest.lru_remove(bh);
|
|
lru_dirty.lru_insert_top(bh);
|
|
dirty_bh.insert(bh);
|
|
}
|
|
if (s != BufferHead::STATE_DIRTY && bh->get_state() == BufferHead::STATE_DIRTY) {
|
|
lru_dirty.lru_remove(bh);
|
|
lru_rest.lru_insert_mid(bh);
|
|
dirty_bh.erase(bh);
|
|
}
|
|
|
|
// set state
|
|
stat_sub(bh);
|
|
bh->set_state(s);
|
|
stat_add(bh);
|
|
}
|
|
|
|
void copy_state(BufferHead *bh1, BufferHead *bh2) {
|
|
set_state(bh2, bh1->get_state());
|
|
}
|
|
|
|
void mark_missing(BufferHead *bh) { set_state(bh, BufferHead::STATE_MISSING); };
|
|
void mark_clean(BufferHead *bh) { set_state(bh, BufferHead::STATE_CLEAN); };
|
|
void mark_rx(BufferHead *bh) { set_state(bh, BufferHead::STATE_RX); };
|
|
void mark_partial(BufferHead *bh) { set_state(bh, BufferHead::STATE_PARTIAL); };
|
|
void mark_tx(BufferHead *bh) { set_state(bh, BufferHead::STATE_TX); };
|
|
void mark_dirty(BufferHead *bh) {
|
|
set_state(bh, BufferHead::STATE_DIRTY);
|
|
bh->set_dirty_stamp(g_clock.now());
|
|
};
|
|
|
|
|
|
// io
|
|
void bh_read(Onode *on, BufferHead *bh, block_t from=0);
|
|
void bh_write(Onode *on, BufferHead *bh, block_t shouldbe=0);
|
|
|
|
bool bh_cancel_read(BufferHead *bh);
|
|
bool bh_cancel_write(BufferHead *bh, version_t cur_epoch);
|
|
|
|
void bh_queue_partial_write(Onode *on, BufferHead *bh);
|
|
void bh_cancel_partial_write(BufferHead *bh);
|
|
|
|
void queue_partial(block_t from, block_t to, map<off_t, bufferlist>& partial, version_t epoch);
|
|
void cancel_partial(block_t from, block_t to, version_t epoch);
|
|
|
|
void add_shadow_partial(block_t from, BufferHead *bh);
|
|
void cancel_shadow_partial(block_t from, BufferHead *bh);
|
|
|
|
void rx_finish(ObjectCache *oc, ioh_t ioh, block_t start, block_t len, block_t diskstart, bufferlist& bl);
|
|
void tx_finish(ObjectCache *oc, ioh_t ioh, block_t start, block_t len, version_t v, version_t e);
|
|
void partial_tx_finish(version_t epoch);
|
|
|
|
friend class C_E_FlushPartial;
|
|
|
|
// bh fun
|
|
BufferHead *split(BufferHead *orig, block_t after);
|
|
};
|
|
|
|
|
|
class C_OC_RxFinish : public BlockDevice::callback {
|
|
Mutex &lock;
|
|
ObjectCache *oc;
|
|
block_t start, length;
|
|
block_t diskstart;
|
|
public:
|
|
bufferlist bl;
|
|
C_OC_RxFinish(Mutex &m, ObjectCache *o, block_t s, block_t l, block_t ds) :
|
|
lock(m), oc(o), start(s), length(l), diskstart(ds) {}
|
|
void finish(ioh_t ioh, int r) {
|
|
oc->bc->rx_finish(oc, ioh, start, length, diskstart, bl);
|
|
}
|
|
};
|
|
|
|
class C_OC_TxFinish : public BlockDevice::callback {
|
|
Mutex &lock;
|
|
ObjectCache *oc;
|
|
block_t start, length;
|
|
version_t version;
|
|
version_t epoch;
|
|
public:
|
|
C_OC_TxFinish(Mutex &m, ObjectCache *o, block_t s, block_t l, version_t v, version_t e) :
|
|
lock(m), oc(o), start(s), length(l), version(v), epoch(e) {}
|
|
void finish(ioh_t ioh, int r) {
|
|
oc->bc->tx_finish(oc, ioh, start, length, version, epoch);
|
|
}
|
|
};
|
|
|
|
class C_OC_PartialTxFinish : public BlockDevice::callback {
|
|
BufferCache *bc;
|
|
version_t epoch;
|
|
public:
|
|
C_OC_PartialTxFinish(BufferCache *b, version_t e) :
|
|
bc(b), epoch(e) {}
|
|
void finish(ioh_t ioh, int r) {
|
|
bc->partial_tx_finish(epoch);
|
|
}
|
|
};
|
|
|
|
|
|
#endif
|