decodeless (previously no-decode) is a
collection of utility libraries for conveniently reading and writing files via
memory mapping. Components can be used individually or combined.
decodeless_header is a general purpose extensible header to make reading and
writing binary files easy. The format definition is just what's in the C++
headers of this git repository. The philosophy is loading the file involves no
serialization or decoding; data and relative structures within can be accessed
directly.
It's up to the individual app to define what is actually stored. This repository
only defines a top level RootHeader which points to a sorted list of
inheritable Headers, identified with a unique ID. The intent is to have one
core header that can be extended with optional headers as the need arises, but
again that's up to the app. Some ideas and recommendations when adding concrete
data to headers:
- Use
offset_ptrandoffset_spanwhich remain valid in different address spaces - Use well defined binary types, e.g. fixed size integers such as
int32_t - Align data types appropriately, e.g. using
decodeless_allocator - Verify binary compatibility on the current system before reading
- Use structures of arrays to allow apps to cherry-pick just the data they need
Example:
// Create a "file"
linear_memory_resource memory(1000);
writeFile(memory, 42);
EXPECT_EQ(memory.size(), 568);
// "Load" the file; could be memory mapped - no time spent decoding or
// deserializing!
auto* root = reinterpret_cast<decodeless::RootHeader*>(memory.data());
// Directly access the file, only reading the parts you need
EXPECT_TRUE(root->binaryCompatible());
auto* appHeader = root->find<AppHeader>();
ASSERT_NE(appHeader, nullptr);
EXPECT_TRUE(
decodeless::Version::binaryCompatible(AppHeader::VersionSupported, appHeader->version));
// The offset_span accessor, data points to an arbitrary location in the
// file, not inside the header
EXPECT_EQ(appHeader->data[0], 42);
EXPECT_EQ(appHeader->data.back(), 42);
...
void writeFile(linear_memory_resource<>& memory, int fillValue) {
// RootHeader must be first
TestRootHeader* rootHeader = decodeless::create<TestRootHeader>(memory);
// Allocate the array of sub-headers
rootHeader->headers = decodeless::createArray<decodeless::offset_ptr<decodeless::Header>>(memory, 1);
// Allocate the app header, its data and populate it
AppHeader* appHeader = decodeless::create<AppHeader>(memory);
appHeader->data = decodeless::createArray<int>(memory, 100);
std::ranges::fill(appHeader->data, fillValue);
// Add the app header the root and sort the array (of one item in this case)
rootHeader->headers[0] = appHeader;
std::ranges::sort(rootHeader->headers, RootHeader::HeaderPtrComp());
}
Issues and pull requests are most welcome, thank you! Note the DCO and MIT LICENSE.