182 lines
6.7 KiB
C++
182 lines
6.7 KiB
C++
/*
|
|
* IPager.h
|
|
*
|
|
* This source file is part of the FoundationDB open source project
|
|
*
|
|
* Copyright 2013-2018 Apple Inc. and the FoundationDB project authors
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#ifndef FDBSERVER_IPAGER_H
|
|
#define FDBSERVER_IPAGER_H
|
|
#pragma once
|
|
|
|
#include "fdbserver/IKeyValueStore.h"
|
|
|
|
#include "flow/flow.h"
|
|
#include "fdbclient/FDBTypes.h"
|
|
|
|
#define REDWOOD_DEBUG 0
|
|
|
|
#define debug_printf_stream stderr
|
|
#define debug_printf_always(...) { fprintf(debug_printf_stream, "%s %f %04d ", g_network->getLocalAddress().toString().c_str(), now(), __LINE__); fprintf(debug_printf_stream, __VA_ARGS__); fflush(debug_printf_stream); }
|
|
|
|
#define debug_printf_noop(...)
|
|
|
|
#if defined(NO_INTELLISENSE)
|
|
#if REDWOOD_DEBUG
|
|
#define debug_printf debug_printf_always
|
|
#else
|
|
#define debug_printf debug_printf_noop
|
|
#endif
|
|
#else
|
|
// To get error-checking on debug_printf statements in IDE
|
|
#define debug_printf printf
|
|
#endif
|
|
|
|
#define BEACON debug_printf_always("HERE\n")
|
|
#define TRACE debug_printf_always("%s: %s line %d %s\n", __FUNCTION__, __FILE__, __LINE__, platform::get_backtrace().c_str());
|
|
|
|
#ifndef VALGRIND
|
|
#define VALGRIND_MAKE_MEM_UNDEFINED(x, y)
|
|
#define VALGRIND_MAKE_MEM_DEFINED(x, y)
|
|
#endif
|
|
|
|
typedef uint32_t LogicalPageID;
|
|
typedef uint32_t PhysicalPageID;
|
|
#define invalidLogicalPageID std::numeric_limits<LogicalPageID>::max()
|
|
|
|
class IPage {
|
|
public:
|
|
IPage() : userData(nullptr) {}
|
|
|
|
virtual uint8_t const* begin() const = 0;
|
|
virtual uint8_t* mutate() = 0;
|
|
|
|
// Must return the same size for all pages created by the same pager instance
|
|
virtual int size() const = 0;
|
|
|
|
StringRef asStringRef() const {
|
|
return StringRef(begin(), size());
|
|
}
|
|
|
|
virtual ~IPage() {
|
|
if(userData != nullptr && userDataDestructor != nullptr) {
|
|
userDataDestructor(userData);
|
|
}
|
|
}
|
|
|
|
virtual void addref() const = 0;
|
|
virtual void delref() const = 0;
|
|
|
|
mutable void *userData;
|
|
mutable void (*userDataDestructor)(void *);
|
|
};
|
|
|
|
class IPagerSnapshot {
|
|
public:
|
|
virtual Future<Reference<const IPage>> getPhysicalPage(LogicalPageID pageID, bool cacheable, bool nohit) = 0;
|
|
virtual Version getVersion() const = 0;
|
|
|
|
virtual Key getMetaKey() const = 0;
|
|
|
|
virtual ~IPagerSnapshot() {}
|
|
|
|
virtual void addref() = 0;
|
|
virtual void delref() = 0;
|
|
};
|
|
|
|
// This API is probably customized to the behavior of DWALPager and probably needs some changes to be more generic.
|
|
class IPager2 : public IClosable {
|
|
public:
|
|
// Returns an IPage that can be passed to writePage. The data in the returned IPage might not be zeroed.
|
|
virtual Reference<IPage> newPageBuffer() = 0;
|
|
|
|
// Returns the usable size of pages returned by the pager (i.e. the size of the page that isn't pager overhead).
|
|
// For a given pager instance, separate calls to this function must return the same value.
|
|
// Only valid to call after recovery is complete.
|
|
virtual int getUsablePageSize() = 0;
|
|
|
|
// Allocate a new page ID for a subsequent write. The page will be considered in-use after the next commit
|
|
// regardless of whether or not it was written to.
|
|
virtual Future<LogicalPageID> newPageID() = 0;
|
|
|
|
// Replace the contents of a page with new data across *all* versions.
|
|
// Existing holders of a page reference for pageID, read from any version,
|
|
// may see the effects of this write.
|
|
virtual void updatePage(LogicalPageID pageID, Reference<IPage> data) = 0;
|
|
|
|
// Try to atomically update the contents of a page as of version v in the next commit.
|
|
// If the pager is unable to do this at this time, it may choose to write the data to a new page ID
|
|
// instead and return the new page ID to the caller. Otherwise the original pageID argument will be returned.
|
|
// If a new page ID is returned, the old page ID will be freed as of version v
|
|
virtual Future<LogicalPageID> atomicUpdatePage(LogicalPageID pageID, Reference<IPage> data, Version v) = 0;
|
|
|
|
// Free pageID to be used again after the commit that moves oldestVersion past v
|
|
virtual void freePage(LogicalPageID pageID, Version v) = 0;
|
|
|
|
// Returns the latest data (regardless of version) for a page by LogicalPageID
|
|
// The data returned will be the later of
|
|
// - the most recent committed atomic
|
|
// - the most recent non-atomic write
|
|
// Cacheable indicates that the page should be added to the page cache (if applicable?) as a result of this read.
|
|
// NoHit indicates that the read should not be considered a cache hit, such as when preloading pages that are
|
|
// considered likely to be needed soon.
|
|
virtual Future<Reference<IPage>> readPage(LogicalPageID pageID, bool cacheable = true, bool noHit = false) = 0;
|
|
|
|
// Get a snapshot of the metakey and all pages as of the version v which must be >= getOldestVersion()
|
|
// Note that snapshots at any version may still see the results of updatePage() calls.
|
|
// The snapshot shall be usable until setOldVersion() is called with a version > v.
|
|
virtual Reference<IPagerSnapshot> getReadSnapshot(Version v) = 0;
|
|
|
|
// Atomically make durable all pending page writes, page frees, and update the metadata string.
|
|
virtual Future<Void> commit() = 0;
|
|
|
|
// Get the latest meta key set or committed
|
|
virtual Key getMetaKey() const = 0;
|
|
|
|
// Set the metakey which will be stored in the next commit
|
|
virtual void setMetaKey(KeyRef metaKey) = 0;
|
|
|
|
// Sets the next commit version
|
|
virtual void setCommitVersion(Version v) = 0;
|
|
|
|
virtual StorageBytes getStorageBytes() = 0;
|
|
|
|
// Count of pages in use by the pager client
|
|
virtual Future<int64_t> getUserPageCount() = 0;
|
|
|
|
// Future returned is ready when pager has been initialized from disk and is ready for reads and writes.
|
|
// It is invalid to call most other functions until init() is ready.
|
|
// TODO: Document further.
|
|
virtual Future<Void> init() = 0;
|
|
|
|
// Returns latest committed version
|
|
virtual Version getLatestVersion() = 0;
|
|
|
|
// Returns the oldest readable version as of the most recent committed version
|
|
virtual Version getOldestVersion() = 0;
|
|
|
|
// Sets the oldest readable version to be put into affect at the next commit.
|
|
// The pager can reuse pages that were freed at a version less than v.
|
|
// If any snapshots are in use at a version less than v, the pager can either forcefully
|
|
// invalidate them or keep their versions around until the snapshots are no longer in use.
|
|
virtual void setOldestVersion(Version v) = 0;
|
|
|
|
protected:
|
|
~IPager2() {} // Destruction should be done using close()/dispose() from the IClosable interface
|
|
};
|
|
|
|
#endif
|