libcryfs/implementations/ondisk/OnDiskBlock.cpp

109 lines
3.0 KiB
C++
Raw Normal View History

2014-12-09 17:19:59 +01:00
#include <cstring>
#include <fstream>
#include <boost/filesystem.hpp>
#include "OnDiskBlock.h"
#include "OnDiskBlockStore.h"
#include "../../utils/FileDoesntExistException.h"
2015-12-14 17:16:54 +01:00
#include <messmer/cpp-utils/data/DataUtils.h>
#include <messmer/cpp-utils/assert/assert.h>
2014-12-09 17:19:59 +01:00
using std::istream;
using std::ostream;
using std::ifstream;
using std::ofstream;
using std::ios;
using cpputils::Data;
using cpputils::make_unique_ref;
using cpputils::unique_ref;
using boost::optional;
using boost::none;
2014-12-09 17:19:59 +01:00
namespace bf = boost::filesystem;
namespace blockstore {
namespace ondisk {
OnDiskBlock::OnDiskBlock(const Key &key, const bf::path &filepath, Data data)
2015-10-17 21:10:26 +02:00
: Block(key), _filepath(filepath), _data(std::move(data)), _dataChanged(false), _mutex() {
2014-12-09 17:19:59 +01:00
}
OnDiskBlock::~OnDiskBlock() {
flush();
2014-12-09 17:19:59 +01:00
}
const void *OnDiskBlock::data() const {
2014-12-09 17:19:59 +01:00
return _data.data();
}
void OnDiskBlock::write(const void *source, uint64_t offset, uint64_t size) {
ASSERT(offset <= _data.size() && offset + size <= _data.size(), "Write outside of valid area"); //Also check offset < _data->size() because of possible overflow in the addition
std::memcpy((uint8_t*)_data.data()+offset, source, size);
_dataChanged = true;
2014-12-09 17:19:59 +01:00
}
size_t OnDiskBlock::size() const {
return _data.size();
}
2015-12-14 17:16:54 +01:00
void OnDiskBlock::resize(size_t newSize) {
_data = cpputils::DataUtils::resize(std::move(_data), newSize);
_dataChanged = true;
}
optional<unique_ref<OnDiskBlock>> OnDiskBlock::LoadFromDisk(const bf::path &rootdir, const Key &key) {
2015-01-24 22:08:41 +01:00
auto filepath = rootdir / key.ToString();
2014-12-09 17:19:59 +01:00
try {
//If it isn't a file, Data::LoadFromFile() would usually also crash. We still need this extra check
//upfront, because Data::LoadFromFile() doesn't crash if we give it the path of a directory
//instead the path of a file.
//TODO Data::LoadFromFile now returns boost::optional. Do we then still need this?
2014-12-09 17:19:59 +01:00
if(!bf::is_regular_file(filepath)) {
return none;
2014-12-09 17:19:59 +01:00
}
boost::optional<Data> data = Data::LoadFromFile(filepath);
if (!data) {
return none;
}
return make_unique_ref<OnDiskBlock>(key, filepath, std::move(*data));
2014-12-09 17:19:59 +01:00
} catch (const FileDoesntExistException &e) {
return none;
2014-12-09 17:19:59 +01:00
}
}
optional<unique_ref<OnDiskBlock>> OnDiskBlock::CreateOnDisk(const bf::path &rootdir, const Key &key, Data data) {
2015-01-24 22:08:41 +01:00
auto filepath = rootdir / key.ToString();
2014-12-09 17:19:59 +01:00
if (bf::exists(filepath)) {
return none;
2014-12-09 17:19:59 +01:00
}
auto block = make_unique_ref<OnDiskBlock>(key, filepath, std::move(data));
block->_storeToDisk();
return std::move(block);
2014-12-09 17:19:59 +01:00
}
2015-02-22 00:29:21 +01:00
void OnDiskBlock::RemoveFromDisk(const bf::path &rootdir, const Key &key) {
auto filepath = rootdir / key.ToString();
ASSERT(bf::is_regular_file(filepath), "Block not found on disk");
2015-02-22 00:29:21 +01:00
bf::remove(filepath);
}
2014-12-09 17:19:59 +01:00
void OnDiskBlock::_fillDataWithZeroes() {
_data.FillWithZeroes();
2015-04-15 14:51:41 +02:00
_dataChanged = true;
2014-12-09 17:19:59 +01:00
}
void OnDiskBlock::_storeToDisk() const {
_data.StoreToFile(_filepath);
}
void OnDiskBlock::flush() {
std::unique_lock<std::mutex> lock(_mutex);
if (_dataChanged) {
_storeToDisk();
_dataChanged = false;
}
2014-12-09 17:19:59 +01:00
}
}
}