//===-- DataExtractorTest.cpp ---------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "gtest/gtest.h" #include "lldb/Utility/DataExtractor.h" using namespace lldb_private; TEST(DataExtractorTest, GetBitfield) { uint8_t buffer[] = {0x01, 0x23, 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); lldb::offset_t offset; offset = 0; ASSERT_EQ(buffer[1], LE.GetMaxU64Bitfield(&offset, sizeof(buffer), 8, 8)); offset = 0; ASSERT_EQ(buffer[1], BE.GetMaxU64Bitfield(&offset, sizeof(buffer), 8, 8)); offset = 0; ASSERT_EQ(static_cast(0xEFCDAB8967452301), LE.GetMaxU64Bitfield(&offset, sizeof(buffer), 64, 0)); offset = 0; ASSERT_EQ(static_cast(0x0123456789ABCDEF), BE.GetMaxU64Bitfield(&offset, sizeof(buffer), 64, 0)); offset = 0; ASSERT_EQ(static_cast(0x01234567), BE.GetMaxU64Bitfield(&offset, sizeof(buffer), 32, 0)); offset = 0; ASSERT_EQ(static_cast(0x012345678), BE.GetMaxU64Bitfield(&offset, sizeof(buffer), 36, 0)); offset = 0; ASSERT_EQ(int8_t(buffer[1]), LE.GetMaxS64Bitfield(&offset, sizeof(buffer), 8, 8)); offset = 0; ASSERT_EQ(int8_t(buffer[1]), BE.GetMaxS64Bitfield(&offset, sizeof(buffer), 8, 8)); offset = 0; ASSERT_EQ(static_cast(0xEFCDAB8967452301), LE.GetMaxS64Bitfield(&offset, sizeof(buffer), 64, 0)); offset = 0; ASSERT_EQ(static_cast(0x0123456789ABCDEF), BE.GetMaxS64Bitfield(&offset, sizeof(buffer), 64, 0)); } TEST(DataExtractorTest, PeekData) { uint8_t buffer[] = {0x01, 0x02, 0x03, 0x04}; DataExtractor E(buffer, sizeof buffer, lldb::eByteOrderLittle, 4); EXPECT_EQ(buffer + 0, E.PeekData(0, 0)); EXPECT_EQ(buffer + 0, E.PeekData(0, 4)); EXPECT_EQ(nullptr, E.PeekData(0, 5)); EXPECT_EQ(buffer + 2, E.PeekData(2, 0)); EXPECT_EQ(buffer + 2, E.PeekData(2, 2)); EXPECT_EQ(nullptr, E.PeekData(2, 3)); EXPECT_EQ(buffer + 4, E.PeekData(4, 0)); EXPECT_EQ(nullptr, E.PeekData(4, 1)); } TEST(DataExtractorTest, GetCStr) { uint8_t buffer[] = {'X', 'f', 'o', 'o', '\0'}; DataExtractor E(buffer, sizeof buffer, lldb::eByteOrderLittle, 4); lldb::offset_t offset = 1; EXPECT_STREQ("foo", E.GetCStr(&offset)); EXPECT_EQ(5U, offset); } TEST(DataExtractorTest, GetCStrEmpty) { uint8_t buffer[] = {'X', '\0'}; DataExtractor E(buffer, sizeof buffer, lldb::eByteOrderLittle, 4); lldb::offset_t offset = 1; EXPECT_STREQ("", E.GetCStr(&offset)); EXPECT_EQ(2U, offset); } TEST(DataExtractorTest, GetCStrUnterminated) { uint8_t buffer[] = {'X', 'f', 'o', 'o'}; DataExtractor E(buffer, sizeof buffer, lldb::eByteOrderLittle, 4); lldb::offset_t offset = 1; EXPECT_EQ(nullptr, E.GetCStr(&offset)); EXPECT_EQ(1U, offset); } TEST(DataExtractorTest, GetCStrAtEnd) { uint8_t buffer[] = {'X'}; DataExtractor E(buffer, sizeof buffer, lldb::eByteOrderLittle, 4); lldb::offset_t offset = 1; EXPECT_EQ(nullptr, E.GetCStr(&offset)); EXPECT_EQ(1U, offset); } TEST(DataExtractorTest, GetCStrAtNullOffset) { uint8_t buffer[] = {'f', 'o', 'o', '\0'}; DataExtractor E(buffer, sizeof buffer, lldb::eByteOrderLittle, 4); lldb::offset_t offset = 0; EXPECT_STREQ("foo", E.GetCStr(&offset)); EXPECT_EQ(4U, offset); } TEST(DataExtractorTest, UncommonAddressSize) { uint8_t buffer[] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}; DataExtractor E2(buffer, sizeof buffer, lldb::eByteOrderLittle, 2); DataExtractor E5(buffer, sizeof buffer, lldb::eByteOrderLittle, 5); DataExtractor E7(buffer, sizeof buffer, lldb::eByteOrderLittle, 7); lldb::offset_t offset; // Test 2-byte addresses (for AVR). offset = 0; EXPECT_EQ(0x0201U, E2.GetMaxU64(&offset, 2)); EXPECT_EQ(2U, offset); offset = 0; EXPECT_EQ(0x0201U, E2.GetAddress(&offset)); EXPECT_EQ(2U, offset); // Test 5-byte addresses. offset = 0; EXPECT_EQ(0x030201U, E5.GetMaxU64(&offset, 3)); EXPECT_EQ(3U, offset); offset = 3; EXPECT_EQ(0x0807060504U, E5.GetAddress(&offset)); EXPECT_EQ(8U, offset); // Test 7-byte addresses. offset = 0; EXPECT_EQ(0x0504030201U, E7.GetMaxU64(&offset, 5)); EXPECT_EQ(5U, offset); offset = 0; EXPECT_EQ(0x07060504030201U, E7.GetAddress(&offset)); EXPECT_EQ(7U, offset); } TEST(DataExtractorTest, GetMaxU64) { uint8_t buffer[] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); lldb::offset_t offset; // Check with the minimum allowed byte size. offset = 0; EXPECT_EQ(0x01U, LE.GetMaxU64(&offset, 1)); EXPECT_EQ(1U, offset); offset = 0; EXPECT_EQ(0x01U, BE.GetMaxU64(&offset, 1)); EXPECT_EQ(1U, offset); // Check with a non-zero offset. offset = 1; EXPECT_EQ(0x0302U, LE.GetMaxU64(&offset, 2)); EXPECT_EQ(3U, offset); offset = 1; EXPECT_EQ(0x0203U, BE.GetMaxU64(&offset, 2)); EXPECT_EQ(3U, offset); // Check with the byte size not being a multiple of 2. offset = 0; EXPECT_EQ(0x07060504030201U, LE.GetMaxU64(&offset, 7)); EXPECT_EQ(7U, offset); offset = 0; EXPECT_EQ(0x01020304050607U, BE.GetMaxU64(&offset, 7)); EXPECT_EQ(7U, offset); // Check with the maximum allowed byte size. offset = 0; EXPECT_EQ(0x0807060504030201U, LE.GetMaxU64(&offset, 8)); EXPECT_EQ(8U, offset); offset = 0; EXPECT_EQ(0x0102030405060708U, BE.GetMaxU64(&offset, 8)); EXPECT_EQ(8U, offset); } TEST(DataExtractorTest, GetMaxS64) { uint8_t buffer[] = {0x01, 0x02, 0x83, 0x04, 0x05, 0x06, 0x07, 0x08}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); lldb::offset_t offset; // Check with the minimum allowed byte size. offset = 0; EXPECT_EQ(0x01, LE.GetMaxS64(&offset, 1)); EXPECT_EQ(1U, offset); offset = 0; EXPECT_EQ(0x01, BE.GetMaxS64(&offset, 1)); EXPECT_EQ(1U, offset); // Check that sign extension works correctly. offset = 0; int64_t value = LE.GetMaxS64(&offset, 3); EXPECT_EQ(0xffffffffff830201U, *reinterpret_cast(&value)); EXPECT_EQ(3U, offset); offset = 2; value = BE.GetMaxS64(&offset, 3); EXPECT_EQ(0xffffffffff830405U, *reinterpret_cast(&value)); EXPECT_EQ(5U, offset); // Check with the maximum allowed byte size. offset = 0; EXPECT_EQ(0x0807060504830201, LE.GetMaxS64(&offset, 8)); EXPECT_EQ(8U, offset); offset = 0; EXPECT_EQ(0x0102830405060708, BE.GetMaxS64(&offset, 8)); EXPECT_EQ(8U, offset); } TEST(DataExtractorTest, GetMaxU64_unchecked) { uint8_t buffer[] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); lldb::offset_t offset; // Check with the minimum allowed byte size. offset = 0; EXPECT_EQ(0x01U, LE.GetMaxU64_unchecked(&offset, 1)); EXPECT_EQ(1U, offset); offset = 0; EXPECT_EQ(0x01U, BE.GetMaxU64_unchecked(&offset, 1)); EXPECT_EQ(1U, offset); // Check with a non-zero offset. offset = 1; EXPECT_EQ(0x0302U, LE.GetMaxU64_unchecked(&offset, 2)); EXPECT_EQ(3U, offset); offset = 1; EXPECT_EQ(0x0203U, BE.GetMaxU64_unchecked(&offset, 2)); EXPECT_EQ(3U, offset); // Check with the byte size not being a multiple of 2. offset = 0; EXPECT_EQ(0x07060504030201U, LE.GetMaxU64_unchecked(&offset, 7)); EXPECT_EQ(7U, offset); offset = 0; EXPECT_EQ(0x01020304050607U, BE.GetMaxU64_unchecked(&offset, 7)); EXPECT_EQ(7U, offset); // Check with the maximum allowed byte size. offset = 0; EXPECT_EQ(0x0807060504030201U, LE.GetMaxU64_unchecked(&offset, 8)); EXPECT_EQ(8U, offset); offset = 0; EXPECT_EQ(0x0102030405060708U, BE.GetMaxU64_unchecked(&offset, 8)); EXPECT_EQ(8U, offset); } TEST(DataExtractorTest, GetSLEB128_bit63) { uint8_t buffer[] = {0xff, 0x80, 0xff, 0x80, 0xff, 0x80, 0xff, 0x80, 0x7f}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); lldb::offset_t offset; int64_t expected = 0b1111111100000001111111000000011111110000000111111100000001111111; offset = 0; EXPECT_EQ(expected, LE.GetSLEB128(&offset)); EXPECT_EQ(9U, offset); offset = 0; EXPECT_EQ(expected, BE.GetSLEB128(&offset)); EXPECT_EQ(9U, offset); } TEST(DataExtractorTest, GetULEB128_bit63) { uint8_t buffer[] = {0xff, 0x80, 0xff, 0x80, 0xff, 0x80, 0xff, 0x80, 0x7f}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); lldb::offset_t offset; uint64_t expected = 0b0111111100000001111111000000011111110000000111111100000001111111; offset = 0; EXPECT_EQ(expected, LE.GetULEB128(&offset)); EXPECT_EQ(9U, offset); offset = 0; EXPECT_EQ(expected, BE.GetULEB128(&offset)); EXPECT_EQ(9U, offset); } TEST(DataExtractorTest, GetFloat) { float expected = 4.0f; lldb::offset_t offset; { uint8_t buffer[] = {0x00, 0x00, 0x80, 0x40}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); offset = 0; EXPECT_DOUBLE_EQ(expected, LE.GetFloat(&offset)); EXPECT_EQ(4U, offset); } { uint8_t buffer[] = {0x40, 0x80, 0x00, 0x00}; DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); offset = 0; EXPECT_DOUBLE_EQ(expected, BE.GetFloat(&offset)); EXPECT_EQ(4U, offset); } } TEST(DataExtractorTest, GetFloatUnaligned) { float expected = 4.0f; lldb::offset_t offset; { uint8_t buffer[] = {0x00, 0x00, 0x00, 0x80, 0x40}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); offset = 1; EXPECT_DOUBLE_EQ(expected, LE.GetFloat(&offset)); EXPECT_EQ(5U, offset); } { uint8_t buffer[] = {0x00, 0x40, 0x80, 0x00, 0x00}; DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); offset = 1; EXPECT_DOUBLE_EQ(expected, BE.GetFloat(&offset)); EXPECT_EQ(5U, offset); } } TEST(DataExtractorTest, GetDouble) { if (sizeof(double) != 8) return; double expected = 4.0f; lldb::offset_t offset; { uint8_t buffer[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x40}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); offset = 0; EXPECT_DOUBLE_EQ(expected, LE.GetDouble(&offset)); EXPECT_EQ(8U, offset); } { uint8_t buffer[] = {0x40, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); offset = 0; EXPECT_DOUBLE_EQ(expected, BE.GetDouble(&offset)); EXPECT_EQ(8U, offset); } } TEST(DataExtractorTest, GetDoubleUnaligned) { if (sizeof(double) != 8) return; float expected = 4.0f; lldb::offset_t offset; { uint8_t buffer[] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x40}; DataExtractor LE(buffer, sizeof(buffer), lldb::eByteOrderLittle, sizeof(void *)); offset = 1; EXPECT_DOUBLE_EQ(expected, LE.GetDouble(&offset)); EXPECT_EQ(9U, offset); } { uint8_t buffer[] = {0x00, 0x40, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; DataExtractor BE(buffer, sizeof(buffer), lldb::eByteOrderBig, sizeof(void *)); offset = 1; EXPECT_DOUBLE_EQ(expected, BE.GetDouble(&offset)); EXPECT_EQ(9U, offset); } }