Reader now reuses the buffer created by UTFDecoder.
This commit is contained in:
parent
9940bc91b9
commit
53b22c93ad
|
@ -12,6 +12,7 @@ import core.stdc.string;
|
||||||
import core.thread;
|
import core.thread;
|
||||||
|
|
||||||
import std.algorithm;
|
import std.algorithm;
|
||||||
|
import std.array;
|
||||||
import std.conv;
|
import std.conv;
|
||||||
import std.exception;
|
import std.exception;
|
||||||
import std.stdio;
|
import std.stdio;
|
||||||
|
@ -64,8 +65,6 @@ class ReaderException : YAMLException
|
||||||
final class Reader
|
final class Reader
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
// Allocated space for buffer_.
|
|
||||||
dchar[] bufferAllocated_ = null;
|
|
||||||
// Buffer of currently loaded characters.
|
// Buffer of currently loaded characters.
|
||||||
dchar[] buffer_ = null;
|
dchar[] buffer_ = null;
|
||||||
// Current position within buffer. Only data after this position can be read.
|
// Current position within buffer. Only data after this position can be read.
|
||||||
|
@ -111,14 +110,14 @@ final class Reader
|
||||||
{
|
{
|
||||||
throw new ReaderException("UTF decoding error: " ~ msg);
|
throw new ReaderException("UTF decoding error: " ~ msg);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@trusted nothrow @nogc ~this()
|
buffer_ = decoder_.decoded;
|
||||||
{
|
|
||||||
// Delete the buffer, if allocated.
|
// The part of buffer excluding trailing zeroes.
|
||||||
if(bufferAllocated_ is null){return;}
|
auto noZeros = buffer_;
|
||||||
free(bufferAllocated_.ptr);
|
while(!noZeros.empty && noZeros.back == '\0') { noZeros.popBack(); }
|
||||||
buffer_ = bufferAllocated_ = null;
|
enforce(printable(noZeros[]),
|
||||||
|
new ReaderException("Special unicode characters are not allowed"));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get character at specified index relative to current position.
|
/// Get character at specified index relative to current position.
|
||||||
|
@ -132,11 +131,6 @@ final class Reader
|
||||||
/// or if invalid data is read.
|
/// or if invalid data is read.
|
||||||
dchar peek(size_t index = 0) @safe
|
dchar peek(size_t index = 0) @safe
|
||||||
{
|
{
|
||||||
if(buffer_.length < bufferOffset_ + index + 1)
|
|
||||||
{
|
|
||||||
updateBuffer(index + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(buffer_.length <= bufferOffset_ + index)
|
if(buffer_.length <= bufferOffset_ + index)
|
||||||
{
|
{
|
||||||
throw new ReaderException("Trying to read past the end of the buffer");
|
throw new ReaderException("Trying to read past the end of the buffer");
|
||||||
|
@ -169,11 +163,6 @@ final class Reader
|
||||||
/// Returns: Slice into the internal buffer or an empty slice if out of bounds.
|
/// Returns: Slice into the internal buffer or an empty slice if out of bounds.
|
||||||
const(dstring) slice(size_t start, size_t end) @trusted
|
const(dstring) slice(size_t start, size_t end) @trusted
|
||||||
{
|
{
|
||||||
if(buffer_.length <= bufferOffset_ + end)
|
|
||||||
{
|
|
||||||
updateBuffer(end);
|
|
||||||
}
|
|
||||||
|
|
||||||
end += bufferOffset_;
|
end += bufferOffset_;
|
||||||
start += bufferOffset_;
|
start += bufferOffset_;
|
||||||
end = min(buffer_.length, end);
|
end = min(buffer_.length, end);
|
||||||
|
@ -199,9 +188,6 @@ final class Reader
|
||||||
/// Params: length = Number or characters to get.
|
/// Params: length = Number or characters to get.
|
||||||
///
|
///
|
||||||
/// Returns: Characters starting at current position.
|
/// Returns: Characters starting at current position.
|
||||||
///
|
|
||||||
/// Throws: ReaderException if trying to read past the end of the buffer
|
|
||||||
/// or if invalid data is read.
|
|
||||||
dstring get(size_t length) @safe
|
dstring get(size_t length) @safe
|
||||||
{
|
{
|
||||||
auto result = prefix(length).idup;
|
auto result = prefix(length).idup;
|
||||||
|
@ -212,16 +198,8 @@ final class Reader
|
||||||
/// Move current position forward.
|
/// Move current position forward.
|
||||||
///
|
///
|
||||||
/// Params: length = Number of characters to move position forward.
|
/// Params: length = Number of characters to move position forward.
|
||||||
///
|
|
||||||
/// Throws: ReaderException if trying to read past the end of the buffer
|
|
||||||
/// or if invalid data is read.
|
|
||||||
void forward(size_t length = 1) @safe
|
void forward(size_t length = 1) @safe
|
||||||
{
|
{
|
||||||
if(buffer_.length <= bufferOffset_ + length + 1)
|
|
||||||
{
|
|
||||||
updateBuffer(length + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
mixin FastCharSearch!"\n\u0085\u2028\u2029"d search;
|
mixin FastCharSearch!"\n\u0085\u2028\u2029"d search;
|
||||||
|
|
||||||
while(length > 0)
|
while(length > 0)
|
||||||
|
@ -254,106 +232,6 @@ final class Reader
|
||||||
|
|
||||||
/// Get encoding of the input buffer.
|
/// Get encoding of the input buffer.
|
||||||
final Encoding encoding() @safe pure nothrow const @nogc { return decoder_.encoding; }
|
final Encoding encoding() @safe pure nothrow const @nogc { return decoder_.encoding; }
|
||||||
|
|
||||||
private:
|
|
||||||
// Update buffer to be able to read length characters after buffer offset.
|
|
||||||
//
|
|
||||||
// If there are not enough characters in the buffer, it will get
|
|
||||||
// as many as possible.
|
|
||||||
//
|
|
||||||
// Params: length = Number of characters we need to read.
|
|
||||||
//
|
|
||||||
// Throws: ReaderException if trying to read past the end of the buffer
|
|
||||||
// or if invalid data is read.
|
|
||||||
void updateBuffer(const size_t length) @trusted
|
|
||||||
{
|
|
||||||
// Get rid of unneeded data in the buffer.
|
|
||||||
if(bufferOffset_ > 0)
|
|
||||||
{
|
|
||||||
const size_t bufferLength = buffer_.length - bufferOffset_;
|
|
||||||
memmove(buffer_.ptr, buffer_.ptr + bufferOffset_,
|
|
||||||
bufferLength * dchar.sizeof);
|
|
||||||
buffer_ = buffer_[0 .. bufferLength];
|
|
||||||
bufferOffset_ = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load chars in batches of at most 1024 bytes (256 chars)
|
|
||||||
while(buffer_.length <= bufferOffset_ + length)
|
|
||||||
{
|
|
||||||
loadChars(512);
|
|
||||||
|
|
||||||
if(decoder_.done)
|
|
||||||
{
|
|
||||||
if(buffer_.length == 0 || buffer_[$ - 1] != '\0')
|
|
||||||
{
|
|
||||||
bufferReserve(buffer_.length + 1);
|
|
||||||
buffer_ = bufferAllocated_[0 .. buffer_.length + 1];
|
|
||||||
buffer_[$ - 1] = '\0';
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load more characters to the buffer.
|
|
||||||
//
|
|
||||||
// Params: chars = Recommended number of characters to load.
|
|
||||||
// More characters might be loaded.
|
|
||||||
// Less will be loaded if not enough available.
|
|
||||||
//
|
|
||||||
// Throws: ReaderException on Unicode decoding error,
|
|
||||||
// if nonprintable characters are detected, or
|
|
||||||
// if there is an error reading from the buffer.
|
|
||||||
//
|
|
||||||
void loadChars(size_t chars) @safe
|
|
||||||
{
|
|
||||||
const oldLength = buffer_.length;
|
|
||||||
const oldPosition = decoder_.position;
|
|
||||||
|
|
||||||
bufferReserve(buffer_.length + chars);
|
|
||||||
buffer_ = bufferAllocated_[0 .. buffer_.length + chars];
|
|
||||||
scope(success)
|
|
||||||
{
|
|
||||||
buffer_ = buffer_[0 .. $ - chars];
|
|
||||||
enforce(printable(buffer_[oldLength .. $]),
|
|
||||||
new ReaderException("Special unicode characters are not allowed"));
|
|
||||||
}
|
|
||||||
|
|
||||||
for(size_t c = 0; chars && !decoder_.done;)
|
|
||||||
{
|
|
||||||
const slice = decoder_.getDChars(chars);
|
|
||||||
buffer_[oldLength + c .. oldLength + c + slice.length] = slice[];
|
|
||||||
c += slice.length;
|
|
||||||
chars -= slice.length;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Code shared by loadEntireFile methods.
|
|
||||||
void loadEntireFile_() @safe
|
|
||||||
{
|
|
||||||
const maxChars = decoder_.maxChars;
|
|
||||||
bufferReserve(maxChars + 1);
|
|
||||||
loadChars(maxChars);
|
|
||||||
|
|
||||||
if(buffer_.length == 0 || buffer_[$ - 1] != '\0')
|
|
||||||
{
|
|
||||||
buffer_ = bufferAllocated_[0 .. buffer_.length + 1];
|
|
||||||
buffer_[$ - 1] = '\0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure there is space for at least capacity characters in bufferAllocated_.
|
|
||||||
void bufferReserve(const size_t capacity) @trusted nothrow @nogc
|
|
||||||
{
|
|
||||||
if(bufferAllocated_ !is null && bufferAllocated_.length >= capacity){return;}
|
|
||||||
|
|
||||||
// Handle first allocation as well as reallocation.
|
|
||||||
auto ptr = bufferAllocated_ !is null
|
|
||||||
? realloc(bufferAllocated_.ptr, capacity * dchar.sizeof)
|
|
||||||
: malloc(capacity * dchar.sizeof);
|
|
||||||
bufferAllocated_ = (cast(dchar*)ptr)[0 .. capacity];
|
|
||||||
buffer_ = bufferAllocated_[0 .. buffer_.length];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -394,8 +272,6 @@ struct UTFDecoder
|
||||||
// Decoded (UTF-32) version of the entire input_. If input is UTF-32, this is
|
// Decoded (UTF-32) version of the entire input_. If input is UTF-32, this is
|
||||||
// just a reference to input_.
|
// just a reference to input_.
|
||||||
dchar[] decoded_;
|
dchar[] decoded_;
|
||||||
// The part of decoded_ that has not yet been read through getDChars().
|
|
||||||
dchar[] unread_;
|
|
||||||
|
|
||||||
// Current error message.
|
// Current error message.
|
||||||
//
|
//
|
||||||
|
@ -435,22 +311,21 @@ struct UTFDecoder
|
||||||
assert(input_.length % 4 == 0,
|
assert(input_.length % 4 == 0,
|
||||||
"UTF-32 buffer size must be a multiple of 4");
|
"UTF-32 buffer size must be a multiple of 4");
|
||||||
// No need to decode anything
|
// No need to decode anything
|
||||||
unread_ = decoded_ = cast(dchar[])input_;
|
decoded_ = cast(dchar[])input_;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
// The buffer must be zero terminated for scanner to detect its end.
|
||||||
|
if(decoded_.empty || decoded_.back() != '\0')
|
||||||
|
{
|
||||||
|
decoded_ ~= cast(dchar)'\0';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get maximum number of characters that might be in the buffer.
|
|
||||||
size_t maxChars() const pure @safe nothrow @nogc { return maxChars_; }
|
|
||||||
|
|
||||||
/// Get encoding we're decoding from.
|
/// Get encoding we're decoding from.
|
||||||
UTFEncoding encoding() const pure @safe nothrow @nogc { return encoding_; }
|
UTFEncoding encoding() const pure @safe nothrow @nogc { return encoding_; }
|
||||||
|
|
||||||
/// Get the current position in buffer.
|
/// Get all decoded characters.
|
||||||
size_t position() @safe pure nothrow const @nogc
|
const(dchar[]) decoded() @safe pure nothrow @nogc { return decoded_; }
|
||||||
{
|
|
||||||
return decoded_.length - unread_.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get the error message and clear it.
|
/// Get the error message and clear it.
|
||||||
string getAndClearErrorMessage() @safe pure nothrow @nogc
|
string getAndClearErrorMessage() @safe pure nothrow @nogc
|
||||||
|
@ -460,24 +335,6 @@ struct UTFDecoder
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Are we done decoding?
|
|
||||||
bool done() const pure @safe nothrow @nogc
|
|
||||||
{
|
|
||||||
return position == decoded_.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get as many characters as possible, but at most maxChars.
|
|
||||||
///
|
|
||||||
/// Returns: A slice with decoded characters.
|
|
||||||
const(dchar[]) getDChars(size_t maxChars = size_t.max) @safe pure nothrow @nogc
|
|
||||||
{
|
|
||||||
const chars = min(maxChars, unread_.length);
|
|
||||||
const result = unread_[0 .. chars];
|
|
||||||
unread_ = unread_[chars .. $];
|
|
||||||
assert(result !is null, "NULL error on a getDChars call without an error");
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Decode input_ if it's encoded as UTF-8 or UTF-16.
|
// Decode input_ if it's encoded as UTF-8 or UTF-16.
|
||||||
//
|
//
|
||||||
|
@ -517,7 +374,6 @@ struct UTFDecoder
|
||||||
{
|
{
|
||||||
assert(false, "Unexpected exception in decode(): " ~ e.msg);
|
assert(false, "Unexpected exception in decode(): " ~ e.msg);
|
||||||
}
|
}
|
||||||
unread_ = decoded_[];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine the end of last UTF-8 or UTF-16 sequence in a raw buffer.
|
// Determine the end of last UTF-8 or UTF-16 sequence in a raw buffer.
|
||||||
|
|
Loading…
Reference in a new issue