mirror of https://github.com/adamdruppe/arsd.git
new module to help work with files
This commit is contained in:
parent
c3a590d0fb
commit
f3fdccb620
|
@ -0,0 +1,178 @@
|
|||
/++
|
||||
A declarative file/stream loader/saver. You define structs with a handful of annotations, this read and writes them to/from files.
|
||||
+/
|
||||
module arsd.declarativeloader;
|
||||
|
||||
import std.range;
|
||||
|
||||
// @VariableLength indicates the value is saved in a MIDI like format
|
||||
// @BigEndian, @LittleEndian
|
||||
// @NumBytes!Field or @NumElements!Field controls length of embedded arrays
|
||||
// @Tagged!Field indicates a tagged union. Each struct within should have @Tag(X) which is a value of Field
|
||||
// @MustBe() causes it to throw if not the given value
|
||||
|
||||
// @NotSaved indicates a struct member that is not actually saved in the file
|
||||
|
||||
enum BigEndian;
|
||||
enum LittleEndian;
|
||||
enum VariableLength;
|
||||
struct NumBytes(alias field) {}
|
||||
struct NumElements(alias field) {}
|
||||
struct Tagged(alias field) {}
|
||||
struct TagStruct(T) { T t; }
|
||||
auto Tag(T)(T t) {
|
||||
return TagStruct!T(t);
|
||||
}
|
||||
enum NotSaved;
|
||||
struct MustBeStruct(T) { T t; }
|
||||
auto MustBe(T)(T t) {
|
||||
return MustBeStruct!T(t);
|
||||
}
|
||||
|
||||
static bool fieldSaved(alias a)() {
|
||||
bool saved;
|
||||
static if(is(typeof(a.offsetof))) {
|
||||
saved = true;
|
||||
static foreach(attr; __traits(getAttributes, a))
|
||||
static if(is(attr == NotSaved))
|
||||
saved = false;
|
||||
}
|
||||
return saved;
|
||||
}
|
||||
|
||||
static bool bigEndian(alias a)(bool def) {
|
||||
bool be = def;
|
||||
static foreach(attr; __traits(getAttributes, a)) {
|
||||
static if(is(attr == BigEndian))
|
||||
be = true;
|
||||
else static if(is(attr == LittleEndian))
|
||||
be = false;
|
||||
}
|
||||
return be;
|
||||
}
|
||||
|
||||
static auto getTag(alias a)() {
|
||||
static foreach(attr; __traits(getAttributes, a)) {
|
||||
static if(is(typeof(attr) == TagStruct!T, T)) {
|
||||
return attr.t;
|
||||
}
|
||||
}
|
||||
assert(0);
|
||||
}
|
||||
|
||||
union N(ty) {
|
||||
ty member;
|
||||
ubyte[ty.sizeof] bytes;
|
||||
}
|
||||
|
||||
// input range of ubytes...
|
||||
int loadFrom(T, Range)(ref T t, auto ref Range r, bool assumeBigEndian = false) {
|
||||
int bytesConsumed;
|
||||
ubyte next() {
|
||||
auto bfr = r.front;
|
||||
r.popFront;
|
||||
bytesConsumed++;
|
||||
return bfr;
|
||||
}
|
||||
|
||||
bool endianness = bigEndian!T(assumeBigEndian);
|
||||
static foreach(memberName; __traits(allMembers, T)) {{
|
||||
static if(is(typeof(__traits(getMember, T, memberName)))) {
|
||||
alias f = __traits(getMember, T, memberName);
|
||||
alias ty = typeof(f);
|
||||
static if(fieldSaved!f) {
|
||||
endianness = bigEndian!f(endianness);
|
||||
// FIXME VariableLength
|
||||
static if(is(ty : ulong) || is(ty : double)) {
|
||||
N!ty n;
|
||||
if(endianness) {
|
||||
foreach(i; 0 .. ty.sizeof) {
|
||||
version(BigEndian)
|
||||
n.bytes[i] = next();
|
||||
else
|
||||
n.bytes[$ - 1 - i] = next();
|
||||
}
|
||||
} else {
|
||||
foreach(i; 0 .. ty.sizeof) {
|
||||
version(BigEndian)
|
||||
n.bytes[$ - 1 - i] = next();
|
||||
else
|
||||
n.bytes[i] = next();
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: MustBe
|
||||
|
||||
__traits(getMember, t, memberName) = n.member;
|
||||
} else static if(is(ty == struct)) {
|
||||
bytesConsumed += loadFrom(__traits(getMember, t, memberName), r, endianness);
|
||||
} else static if(is(ty == union)) {
|
||||
static foreach(attr; __traits(getAttributes, ty))
|
||||
static if(is(attr == Tagged!Field, alias Field))
|
||||
enum tagField = __traits(identifier, Field);
|
||||
static assert(is(typeof(tagField)), "Unions need a Tagged UDA on the union type (not the member) indicating the field that identifies the union");
|
||||
|
||||
auto tag = __traits(getMember, t, tagField);
|
||||
// find the child of the union matching the tag...
|
||||
static foreach(um; __traits(allMembers, ty)) {
|
||||
if(tag == getTag!(__traits(getMember, ty, um))) {
|
||||
bytesConsumed += loadFrom(__traits(getMember, __traits(getMember, t, memberName), um), r, endianness);
|
||||
}
|
||||
}
|
||||
} else static if(is(ty == E[], E)) {
|
||||
static foreach(attr; __traits(getAttributes, f)) {
|
||||
static if(is(attr == NumBytes!Field, alias Field))
|
||||
ulong numBytesRemaining = __traits(getMember, t, __traits(identifier, Field));
|
||||
else static if(is(attr == NumElements!Field, alias Field)) {
|
||||
ulong numElementsRemaining = __traits(getMember, t, __traits(identifier, Field));
|
||||
}
|
||||
}
|
||||
|
||||
static if(is(typeof(numBytesRemaining))) {
|
||||
static if(is(E : const(ubyte)) || is(E : const(char))) {
|
||||
while(numBytesRemaining) {
|
||||
__traits(getMember, t, memberName) ~= next;
|
||||
numBytesRemaining--;
|
||||
}
|
||||
} else {
|
||||
while(numBytesRemaining) {
|
||||
E piece;
|
||||
auto by = loadFrom(e, r, endianness);
|
||||
numBytesRemaining -= by;
|
||||
bytesConsumed += by;
|
||||
__traits(getMember, t, memberName) ~= piece;
|
||||
}
|
||||
}
|
||||
} else static if(is(typeof(numElementsRemaining))) {
|
||||
static if(is(E : const(ubyte)) || is(E : const(char))) {
|
||||
while(numElementsRemaining) {
|
||||
__traits(getMember, t, memberName) ~= next;
|
||||
numElementsRemaining--;
|
||||
}
|
||||
} else static if(is(E : const(ushort))) {
|
||||
while(numElementsRemaining) {
|
||||
ushort n;
|
||||
n = next << 8;
|
||||
n |= next;
|
||||
// FIXME all of this filth
|
||||
__traits(getMember, t, memberName) ~= n;
|
||||
numElementsRemaining--;
|
||||
}
|
||||
} else {
|
||||
while(numElementsRemaining) {
|
||||
E piece;
|
||||
auto by = loadFrom(piece, r, endianness);
|
||||
numElementsRemaining--;
|
||||
bytesConsumed += by;
|
||||
__traits(getMember, t, memberName) ~= piece;
|
||||
}
|
||||
}
|
||||
} else static assert(0, "no way to identify length... " ~ memberName);
|
||||
|
||||
} else static assert(0, ty.stringof);
|
||||
}
|
||||
}
|
||||
}}
|
||||
|
||||
return bytesConsumed;
|
||||
}
|
Loading…
Reference in New Issue