mirror of
https://github.com/RGBCube/serenity
synced 2025-07-27 14:47:46 +00:00
AK: Move the LEB128 logic to AK and make it usable with InputStream
This commit is contained in:
parent
8cc6372b14
commit
48260b5054
2 changed files with 88 additions and 51 deletions
85
AK/LEB128.h
Normal file
85
AK/LEB128.h
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2020-2021, the SerenityOS developers.
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/Stream.h>
|
||||||
|
#include <AK/Types.h>
|
||||||
|
|
||||||
|
namespace AK {
|
||||||
|
|
||||||
|
struct LEB128 {
|
||||||
|
template<typename StreamT>
|
||||||
|
static bool read_unsigned(StreamT& stream, size_t& result)
|
||||||
|
{
|
||||||
|
[[maybe_unused]] size_t backup_offset = 0;
|
||||||
|
if constexpr (requires { stream.offset(); })
|
||||||
|
backup_offset = stream.offset();
|
||||||
|
InputStream& input_stream { stream };
|
||||||
|
|
||||||
|
result = 0;
|
||||||
|
size_t num_bytes = 0;
|
||||||
|
while (true) {
|
||||||
|
if (input_stream.unreliable_eof()) {
|
||||||
|
if constexpr (requires { stream.seek(backup_offset); })
|
||||||
|
stream.seek(backup_offset);
|
||||||
|
input_stream.set_fatal_error();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
u8 byte = 0;
|
||||||
|
input_stream >> byte;
|
||||||
|
if (input_stream.has_any_error())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
result = (result) | (static_cast<size_t>(byte & ~(1 << 7)) << (num_bytes * 7));
|
||||||
|
if (!(byte & (1 << 7)))
|
||||||
|
break;
|
||||||
|
++num_bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename StreamT>
|
||||||
|
static bool read_signed(StreamT& stream, ssize_t& result)
|
||||||
|
{
|
||||||
|
[[maybe_unused]] size_t backup_offset = 0;
|
||||||
|
if constexpr (requires { stream.offset(); })
|
||||||
|
backup_offset = stream.offset();
|
||||||
|
InputStream& input_stream { stream };
|
||||||
|
|
||||||
|
result = 0;
|
||||||
|
size_t num_bytes = 0;
|
||||||
|
u8 byte = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
if (input_stream.unreliable_eof()) {
|
||||||
|
if constexpr (requires { stream.seek(backup_offset); })
|
||||||
|
stream.seek(backup_offset);
|
||||||
|
input_stream.set_fatal_error();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
input_stream >> byte;
|
||||||
|
if (input_stream.has_any_error())
|
||||||
|
return false;
|
||||||
|
result = (result) | (static_cast<size_t>(byte & ~(1 << 7)) << (num_bytes * 7));
|
||||||
|
++num_bytes;
|
||||||
|
} while (byte & (1 << 7));
|
||||||
|
|
||||||
|
if (num_bytes * 7 < sizeof(size_t) * 4 && (byte & 0x40)) {
|
||||||
|
// sign extend
|
||||||
|
result |= ((size_t)(-1) << (num_bytes * 7));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
using AK::LEB128;
|
|
@ -7,6 +7,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <AK/ByteBuffer.h>
|
#include <AK/ByteBuffer.h>
|
||||||
|
#include <AK/LEB128.h>
|
||||||
#include <AK/MemMem.h>
|
#include <AK/MemMem.h>
|
||||||
#include <AK/Stream.h>
|
#include <AK/Stream.h>
|
||||||
#include <AK/Vector.h>
|
#include <AK/Vector.h>
|
||||||
|
@ -73,58 +74,9 @@ public:
|
||||||
return m_bytes[m_offset];
|
return m_bytes[m_offset];
|
||||||
}
|
}
|
||||||
|
|
||||||
bool read_LEB128_unsigned(size_t& result)
|
bool read_LEB128_unsigned(size_t& result) { return LEB128::read_unsigned(*this, result); }
|
||||||
{
|
|
||||||
const auto backup = m_offset;
|
|
||||||
|
|
||||||
result = 0;
|
bool read_LEB128_signed(ssize_t& result) { return LEB128::read_signed(*this, result); }
|
||||||
size_t num_bytes = 0;
|
|
||||||
while (true) {
|
|
||||||
if (eof()) {
|
|
||||||
m_offset = backup;
|
|
||||||
set_recoverable_error();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const u8 byte = m_bytes[m_offset];
|
|
||||||
result = (result) | (static_cast<size_t>(byte & ~(1 << 7)) << (num_bytes * 7));
|
|
||||||
++m_offset;
|
|
||||||
if (!(byte & (1 << 7)))
|
|
||||||
break;
|
|
||||||
++num_bytes;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool read_LEB128_signed(ssize_t& result)
|
|
||||||
{
|
|
||||||
const auto backup = m_offset;
|
|
||||||
|
|
||||||
result = 0;
|
|
||||||
size_t num_bytes = 0;
|
|
||||||
u8 byte = 0;
|
|
||||||
|
|
||||||
do {
|
|
||||||
if (eof()) {
|
|
||||||
m_offset = backup;
|
|
||||||
set_recoverable_error();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
byte = m_bytes[m_offset];
|
|
||||||
result = (result) | (static_cast<size_t>(byte & ~(1 << 7)) << (num_bytes * 7));
|
|
||||||
++m_offset;
|
|
||||||
++num_bytes;
|
|
||||||
} while (byte & (1 << 7));
|
|
||||||
|
|
||||||
if (num_bytes * 7 < sizeof(size_t) * 4 && (byte & 0x40)) {
|
|
||||||
// sign extend
|
|
||||||
result |= ((size_t)(-1) << (num_bytes * 7));
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
ReadonlyBytes bytes() const { return m_bytes; }
|
ReadonlyBytes bytes() const { return m_bytes; }
|
||||||
size_t offset() const { return m_offset; }
|
size_t offset() const { return m_offset; }
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue