blob: eae2ac78c8c5a10c89884fa395726748f8b92eb0 [file] [log] [blame]
// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/gwp_asan/common/pack_stack_trace.h"
#include "base/compiler_specific.h"
namespace gwp_asan {
namespace internal {
namespace {
// Encode variable-length integer to |out|, returns 0 if there was not enough
// space to finish writing it or the length of the encoded integer otherwise.
size_t VarIntEncode(uintptr_t value, uint8_t* out, size_t out_len) {
for (size_t i = 0; i < out_len; i++) {
UNSAFE_TODO(out[i]) = value & 0x7f;
value >>= 7;
if (!value)
return i + 1;
UNSAFE_TODO(out[i]) |= 0x80;
}
return 0;
}
// Decode a variable-length integer to |out|, returns 0 if reading it failed or
// the length of the decoded integer otherwise.
size_t VarIntDecode(const uint8_t* in, size_t in_len, uintptr_t* out) {
uintptr_t result = 0;
size_t shift = 0;
for (size_t i = 0; i < in_len; i++) {
result |= static_cast<uintptr_t>(UNSAFE_TODO(in[i]) & 0x7f) << shift;
if (UNSAFE_TODO(in[i]) < 0x80) {
*out = result;
return i + 1;
}
shift += 7;
// Disallow overflowing the range of the output integer.
if (shift >= sizeof(uintptr_t) * 8)
return 0;
}
return 0;
}
uintptr_t ZigzagEncode(uintptr_t value) {
uintptr_t encoded = value << 1;
if (static_cast<intptr_t>(value) >= 0)
return encoded;
return ~encoded;
}
uintptr_t ZigzagDecode(uintptr_t value) {
uintptr_t decoded = value >> 1;
if (!(value & 1))
return decoded;
return ~decoded;
}
} // namespace
size_t Pack(const uintptr_t* unpacked,
size_t unpacked_size,
uint8_t* packed,
size_t packed_max_size) {
size_t idx = 0;
for (size_t cur_depth = 0; cur_depth < unpacked_size; cur_depth++) {
uintptr_t diff = UNSAFE_TODO(unpacked[cur_depth]);
if (cur_depth > 0)
diff -= UNSAFE_TODO(unpacked[cur_depth - 1]);
size_t encoded_len = VarIntEncode(
ZigzagEncode(diff), UNSAFE_TODO(packed + idx), packed_max_size - idx);
if (!encoded_len)
break;
idx += encoded_len;
}
return idx;
}
size_t Unpack(const uint8_t* packed,
size_t packed_size,
uintptr_t* unpacked,
size_t unpacked_max_size) {
size_t cur_depth;
size_t idx = 0;
for (cur_depth = 0; cur_depth < unpacked_max_size; cur_depth++) {
uintptr_t encoded_diff;
size_t decoded_len = VarIntDecode(UNSAFE_TODO(packed + idx),
packed_size - idx, &encoded_diff);
if (!decoded_len)
break;
idx += decoded_len;
UNSAFE_TODO(unpacked[cur_depth]) = ZigzagDecode(encoded_diff);
if (cur_depth > 0)
UNSAFE_TODO(unpacked[cur_depth]) += UNSAFE_TODO(unpacked[cur_depth - 1]);
}
if (idx != packed_size && cur_depth != unpacked_max_size)
return 0;
return cur_depth;
}
} // namespace internal
} // namespace gwp_asan