X-Git-Url: https://git.saurik.com/cydia.git/blobdiff_plain/ffbb3bd57668b3733a13f00d43760d8b9abfd887..069ee7886cc06b3db4a8d67f2a03a2f4e8d30fa6:/Cytore.hpp diff --git a/Cytore.hpp b/Cytore.hpp index db400459..c0bad09b 100644 --- a/Cytore.hpp +++ b/Cytore.hpp @@ -50,10 +50,7 @@ struct Header { uint32_t version_; uint32_t size_; uint32_t reserved_; -}; - -struct Block { -}; +} _packed; template class Offset { @@ -83,7 +80,11 @@ class Offset { bool IsNull() const { return offset_ == 0; } -}; +} _packed; + +struct Block { + Cytore::Offset reserved_; +} _packed; template static _finline Type_ Round(Type_ value, Type_ size) { @@ -131,17 +132,23 @@ class File { size_t extend(size - before); void *data(mmap(NULL, extend, PROT_READ | PROT_WRITE, MAP_FILE | MAP_SHARED, file_, before)); + _assert(data != MAP_FAILED); uint8_t *bytes(reinterpret_cast(data)); - maps_.push_back(Mapping_(bytes,extend)); + maps_.push_back(Mapping_(bytes, extend)); for (size_t i(0); i != extend >> Shift_; ++i) blocks_.push_back(bytes + Block_ * i); } - void Truncate_(size_t capacity) { + bool Truncate_(size_t capacity) { capacity = Round(capacity, Block_); - ftruncate(file_, capacity); + + int error(ftruncate(file_, capacity)); + if (error != 0) + return false; + Map_(capacity); + return true; } public: @@ -171,38 +178,56 @@ class File { return blocks_.size() * Block_; } - void Open(const char *path) { + void Open(const char *path) { open: _assert(file_ == -1); file_ = open(path, O_RDWR | O_CREAT | O_EXLOCK, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH); + _assert(file_ != -1); struct stat stat; - fstat(file_, &stat); + _assert(fstat(file_, &stat) == 0); size_t core(sizeof(Header) + sizeof(Base_)); size_t size(stat.st_size); if (size == 0) { - Truncate_(core); + if (!Truncate_(core)) { + unlink(path); + _assert(false); + } + Header_().magic_ = Magic; Size_() = core; + } else if (size < core) { + close(file_); + file_ = -1; + unlink(path); + goto open; } else { - _assert(size >= core); - Truncate_(size); + // XXX: this involves an unneccessary call to ftruncate() + _assert(Truncate_(size)); _assert(Header_().magic_ == Magic); _assert(Header_().version_ == 0); } } - void Reserve(size_t capacity) { + bool Reserve(size_t capacity) { if (capacity <= Capacity()) - return; + return true; + + uint8_t *block(blocks_.back()); blocks_.pop_back(); - Truncate_(capacity); + + if (Truncate_(capacity)) + return true; + else { + blocks_.push_back(block); + return false; + } } template Target_ &Get(uint32_t offset) { - return *reinterpret_cast(blocks_[offset >> Shift_] + (offset & Mask_)); + return *reinterpret_cast(offset == 0 ? NULL : blocks_[offset >> Shift_] + (offset & Mask_)); } template @@ -218,9 +243,15 @@ class File { Offset New(size_t extra = 0) { size_t size(sizeof(Target_) + extra); size = Round(size, sizeof(uintptr_t)); - Reserve(Size_() + size); - uint32_t offset(Size_()); - Size_() += size; + + uint32_t offset; + if (!Reserve(Size_() + size)) + offset = 0; + else { + offset = Size_(); + Size_() += size; + } + return Offset(offset); } };