omg use after free bug was there

This commit is contained in:
Adam D. Ruppe 2024-09-09 21:58:13 -04:00
parent 701fdf5e25
commit 4f2b94e790
1 changed files with 34 additions and 6 deletions

40
dom.d
View File

@ -4453,17 +4453,25 @@ private struct InternalAttribute {
private char[0] chars; private char[0] chars;
// this really should be immutable tbh // this really should be immutable tbh
inout(char)[] key() inout { inout(char)[] key() inout return {
return chars.ptr[0 .. keyLength]; return chars.ptr[0 .. keyLength];
} }
inout(char)[] value() inout { inout(char)[] value() inout return {
return chars.ptr[keyLength .. totalLength]; return chars.ptr[keyLength .. totalLength];
} }
static InternalAttribute* make(in char[] key, in char[] value) { static InternalAttribute* make(in char[] key, in char[] value) {
auto data = new ubyte[](InternalAttribute.sizeof + key.length + value.length); // old code was
auto obj = cast(InternalAttribute*) data.ptr; //auto data = new ubyte[](InternalAttribute.sizeof + key.length + value.length);
//GC.addRange(data.ptr, data.length); // MUST add the range to scan it!
import core.memory;
// but this code is a bit better, notice we did NOT set the NO_SCAN attribute because of the presence of the next pointer
// (this can sometimes be a pessimization over the separate strings but meh, most of these attributes are supposed to be small)
auto obj = cast(InternalAttribute*) GC.calloc(InternalAttribute.sizeof + key.length + value.length);
// assert(key.length > 0);
obj.totalLength = cast(uint) (key.length + value.length); obj.totalLength = cast(uint) (key.length + value.length);
obj.keyLength = cast(ushort) key.length; obj.keyLength = cast(ushort) key.length;
@ -4477,6 +4485,8 @@ private struct InternalAttribute {
return obj; return obj;
} }
// FIXME: disable default ctor and op new
} }
import core.exception; import core.exception;
@ -4484,11 +4494,22 @@ import core.exception;
struct AttributesHolder { struct AttributesHolder {
private @system InternalAttribute* attributes; private @system InternalAttribute* attributes;
/+
invariant() {
const(InternalAttribute)* wtf = attributes;
while(wtf) {
assert(wtf != cast(void*) 1);
assert(wtf.keyLength != 0);
import std.stdio; writeln(wtf.key, "=", wtf.value);
wtf = wtf.next;
}
}
+/
/+ /+
It is legal to do foo["key", "default"] to call it with no error... It is legal to do foo["key", "default"] to call it with no error...
+/ +/
string opIndex(scope const char[] key) const { string opIndex(scope const char[] key) const {
auto thing = attributes;
auto found = find(key); auto found = find(key);
if(found is null) if(found is null)
throw new RangeError(key.idup); // FIXME throw new RangeError(key.idup); // FIXME
@ -4522,6 +4543,7 @@ struct AttributesHolder {
private inout(InternalAttribute)* find(scope const char[] key) inout @trusted { private inout(InternalAttribute)* find(scope const char[] key) inout @trusted {
inout(InternalAttribute)* current = attributes; inout(InternalAttribute)* current = attributes;
while(current) { while(current) {
// assert(current > cast(void*) 1);
if(current.key == key) if(current.key == key)
return current; return current;
current = current.next; current = current.next;
@ -4546,9 +4568,11 @@ struct AttributesHolder {
attributes = current.next; attributes = current.next;
else else
previous.next = current.next; previous.next = current.next;
// assert(previous.next != cast(void*) 1);
// assert(attributes != cast(void*) 1);
} }
void opIndexAssign(scope const char[] value, scope const char[] key) { void opIndexAssign(scope const char[] value, scope const char[] key) @trusted {
if(attributes is null) { if(attributes is null) {
attributes = InternalAttribute.make(key, value); attributes = InternalAttribute.make(key, value);
return; return;
@ -4560,6 +4584,8 @@ struct AttributesHolder {
auto replacement = InternalAttribute.make(key, value); auto replacement = InternalAttribute.make(key, value);
attributes = replacement; attributes = replacement;
replacement.next = current.next; replacement.next = current.next;
// assert(replacement.next != cast(void*) 1);
// assert(attributes != cast(void*) 1);
} }
return; return;
} }
@ -4578,6 +4604,8 @@ struct AttributesHolder {
if(current.next !is null) if(current.next !is null)
replacement.next = current.next.next; replacement.next = current.next.next;
current.next = replacement; current.next = replacement;
// assert(current.next != cast(void*) 1);
// assert(replacement.next != cast(void*) 1);
} }
int opApply(int delegate(string key, string value) dg) const @trusted { int opApply(int delegate(string key, string value) dg) const @trusted {