Add readahead to VariantParser

Adds a readahead buffer to VariantParser, to prevent large numbers of freads for single bytes, which is inefficient.
This commit is contained in:
lawnjelly 2022-08-30 11:42:36 +01:00 committed by Relintai
parent 607964e973
commit 3d1640eb70
2 changed files with 84 additions and 28 deletions

View File

@ -35,36 +35,77 @@
#include "core/os/keyboard.h" #include "core/os/keyboard.h"
#include "core/string/string_buffer.h" #include "core/string/string_buffer.h"
CharType VariantParser::StreamFile::get_char() { CharType VariantParser::Stream::get_char() {
return f->get_8(); // is within buffer?
if (readahead_pointer < readahead_filled) {
return readahead_buffer[readahead_pointer++];
}
// attempt to readahead
readahead_filled = _read_buffer(readahead_buffer, READAHEAD_SIZE);
if (readahead_filled) {
readahead_pointer = 0;
} else {
// EOF
readahead_pointer = 1;
eof = true;
return 0;
}
return get_char();
}
uint32_t VariantParser::StreamFile::_read_buffer(CharType *p_buffer, uint32_t p_num_chars) {
// The buffer is assumed to include at least one character (for null terminator)
ERR_FAIL_COND_V(!p_num_chars, 0);
uint8_t *temp = (uint8_t *)alloca(p_num_chars);
uint64_t num_read = f->get_buffer(temp, p_num_chars);
ERR_FAIL_COND_V(num_read == UINT64_MAX, 0);
// translate to wchar
for (uint32_t n = 0; n < num_read; n++) {
p_buffer[n] = temp[n];
}
// could be less than p_num_chars, or zero
return num_read;
} }
bool VariantParser::StreamFile::is_utf8() const { bool VariantParser::StreamFile::is_utf8() const {
return true; return true;
} }
bool VariantParser::StreamFile::is_eof() const {
return f->eof_reached();
}
CharType VariantParser::StreamString::get_char() { uint32_t VariantParser::StreamString::_read_buffer(CharType *p_buffer, uint32_t p_num_chars) {
if (pos > s.length()) { // The buffer is assumed to include at least one character (for null terminator)
return 0; ERR_FAIL_COND_V(!p_num_chars, 0);
} else if (pos == s.length()) {
// You need to try to read again when you have reached the end for EOF to be reported, int available = MAX(s.length() - pos, 0);
// so this works the same as files (like StreamFile does) if (available >= (int)p_num_chars) {
pos++; const CharType *src = s.ptr();
return 0; src += pos;
} else { memcpy(p_buffer, src, p_num_chars * sizeof(CharType));
return s[pos++]; pos += p_num_chars;
return p_num_chars;
} }
// going to reach EOF
if (available) {
const CharType *src = s.ptr();
src += pos;
memcpy(p_buffer, src, available * sizeof(CharType));
pos += available;
}
// add a zero
p_buffer[available] = 0;
return available;
} }
bool VariantParser::StreamString::is_utf8() const { bool VariantParser::StreamString::is_utf8() const {
return false; return false;
} }
bool VariantParser::StreamString::is_eof() const {
return pos > s.length();
}
///////////////////////////////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////////////////////////////

View File

@ -37,35 +37,50 @@
class VariantParser { class VariantParser {
public: public:
struct Stream { struct Stream {
virtual CharType get_char() = 0; private:
virtual bool is_utf8() const = 0; enum { READAHEAD_SIZE = 2048 };
virtual bool is_eof() const = 0; CharType readahead_buffer[READAHEAD_SIZE];
uint32_t readahead_pointer = 0;
uint32_t readahead_filled = 0;
bool eof = false;
protected:
virtual uint32_t _read_buffer(CharType *p_buffer, uint32_t p_num_chars) = 0;
public:
CharType saved; CharType saved;
CharType get_char();
virtual bool is_utf8() const = 0;
bool is_eof() const { return eof; }
Stream() : Stream() :
saved(0) {} saved(0) {}
virtual ~Stream() {} virtual ~Stream() {}
}; };
struct StreamFile : public Stream { struct StreamFile : public Stream {
protected:
virtual uint32_t _read_buffer(CharType *p_buffer, uint32_t p_num_chars);
public:
FileAccess *f; FileAccess *f;
virtual CharType get_char();
virtual bool is_utf8() const; virtual bool is_utf8() const;
virtual bool is_eof() const;
StreamFile() { f = nullptr; } StreamFile() { f = nullptr; }
}; };
struct StreamString : public Stream { struct StreamString : public Stream {
String s; private:
int pos; int pos;
virtual CharType get_char(); protected:
virtual bool is_utf8() const; virtual uint32_t _read_buffer(CharType *p_buffer, uint32_t p_num_chars);
virtual bool is_eof() const;
public:
String s;
virtual bool is_utf8() const;
StreamString() { pos = 0; } StreamString() { pos = 0; }
}; };