blob: 9207c2d7c28d6e875e35223c8f963edd855f2681 [file] [log] [blame] [edit]
#include "buffer.hpp"
#include "data_interface_mock.hpp"
#include <boost/endian/arithmetic.hpp>
#include <boost/endian/conversion.hpp>
#include <algorithm>
#include <array>
#include <cstdint>
#include <memory>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
namespace bios_bmc_smm_error_logger
{
namespace
{
using ::testing::_;
using ::testing::ElementsAreArray;
using ::testing::InSequence;
using ::testing::Return;
class BufferTest : public ::testing::Test
{
protected:
BufferTest() :
dataInterfaceMock(std::make_unique<DataInterfaceMock>()),
dataInterfaceMockPtr(dataInterfaceMock.get())
{
bufferImpl = std::make_unique<BufferImpl>(std::move(dataInterfaceMock));
testInitializationHeader.bmcInterfaceVersion = testBmcInterfaceVersion;
testInitializationHeader.queueSize = testQueueSize;
testInitializationHeader.ueRegionSize = testUeRegionSize;
std::transform(testMagicNumber.begin(), testMagicNumber.end(),
testInitializationHeader.magicNumber.begin(),
[](uint32_t number) -> little_uint32_t {
return boost::endian::native_to_little(number);
});
}
~BufferTest() override = default;
// CircularBufferHeader size is 0x30, ensure the test region is bigger
static constexpr size_t testRegionSize = 0x200;
static constexpr uint32_t testBmcInterfaceVersion = 123;
static constexpr uint32_t testQueueSize = 0x200;
static constexpr uint16_t testUeRegionSize = 0x50;
static constexpr std::array<uint32_t, 4> testMagicNumber = {
0x12345678, 0x22345678, 0x32345678, 0x42345678};
static constexpr size_t bufferHeaderSize =
sizeof(struct CircularBufferHeader);
struct CircularBufferHeader testInitializationHeader{};
std::unique_ptr<DataInterfaceMock> dataInterfaceMock;
DataInterfaceMock* dataInterfaceMockPtr;
std::unique_ptr<BufferImpl> bufferImpl;
};
TEST_F(BufferTest, BufferInitializeEraseFail)
{
InSequence s;
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
EXPECT_THROW(
try {
// Test too big of a proposed buffer compared to the memori size
uint16_t bigQueueSize = 0x201;
uint16_t bigUeRegionSize = 0x50;
bufferImpl->initialize(testBmcInterfaceVersion, bigQueueSize,
bigUeRegionSize, testMagicNumber);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[initialize] Proposed queue size '513' is bigger than the BMC's allocated MMIO region of '512'");
throw;
},
std::runtime_error);
EXPECT_NE(bufferImpl->getCachedBufferHeader(), testInitializationHeader);
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
const std::vector<uint8_t> emptyArray(testQueueSize, 0);
// Return a smaller write than the intended testRegionSize to test the error
EXPECT_CALL(*dataInterfaceMockPtr, write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(testQueueSize - 1));
EXPECT_THROW(
try {
bufferImpl->initialize(testBmcInterfaceVersion, testQueueSize,
testUeRegionSize, testMagicNumber);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(), "[initialize] Only erased '511'");
throw;
},
std::runtime_error);
EXPECT_NE(bufferImpl->getCachedBufferHeader(), testInitializationHeader);
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
EXPECT_CALL(*dataInterfaceMockPtr, write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(testQueueSize));
// Return a smaller write than the intended initializationHeader to test the
// error
EXPECT_CALL(*dataInterfaceMockPtr, write(0, _)).WillOnce(Return(0));
EXPECT_THROW(
try {
bufferImpl->initialize(testBmcInterfaceVersion, testQueueSize,
testUeRegionSize, testMagicNumber);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(),
"[initialize] Only wrote '0' bytes of the header");
throw;
},
std::runtime_error);
EXPECT_NE(bufferImpl->getCachedBufferHeader(), testInitializationHeader);
}
TEST_F(BufferTest, BufferInitializePass)
{
InSequence s;
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
const std::vector<uint8_t> emptyArray(testQueueSize, 0);
EXPECT_CALL(*dataInterfaceMockPtr, write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(testQueueSize));
uint8_t* testInitializationHeaderPtr =
reinterpret_cast<uint8_t*>(&testInitializationHeader);
EXPECT_CALL(*dataInterfaceMockPtr,
write(0, ElementsAreArray(testInitializationHeaderPtr,
bufferHeaderSize)))
.WillOnce(Return(bufferHeaderSize));
EXPECT_NO_THROW(
bufferImpl->initialize(testBmcInterfaceVersion, testQueueSize,
testUeRegionSize, testMagicNumber));
EXPECT_EQ(bufferImpl->getCachedBufferHeader(), testInitializationHeader);
}
TEST_F(BufferTest, BufferHeaderReadFail)
{
std::vector<std::uint8_t> testBytesRead{};
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(testBytesRead));
EXPECT_THROW(
try {
bufferImpl->readBufferHeader();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(),
"Buffer header read only read '0', expected '48'");
throw;
},
std::runtime_error);
}
TEST_F(BufferTest, BufferHeaderReadPass)
{
uint8_t* testInitializationHeaderPtr =
reinterpret_cast<uint8_t*>(&testInitializationHeader);
std::vector<uint8_t> testInitializationHeaderVector(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize);
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(testInitializationHeaderVector));
EXPECT_NO_THROW(bufferImpl->readBufferHeader());
EXPECT_EQ(bufferImpl->getCachedBufferHeader(), testInitializationHeader);
}
TEST_F(BufferTest, BufferUpdateReadPtrFail)
{
// Return write size that is not 2 which is sizeof(little_uint16_t)
constexpr size_t wrongWriteSize = 1;
EXPECT_CALL(*dataInterfaceMockPtr, write(_, _))
.WillOnce(Return(wrongWriteSize));
EXPECT_THROW(
try {
bufferImpl->updateReadPtr(0);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[updateReadPtr] Wrote '1' bytes, instead of expected '3'");
throw;
},
std::runtime_error);
}
TEST_F(BufferTest, BufferUpdateReadPtrPass)
{
constexpr size_t expectedWriteSize = 3;
constexpr uint8_t expectedBmcReadPtrOffset = 0x21;
// Check that we truncate the highest 24bits
const uint32_t testNewReadPtr = 0x99881234;
const std::vector<uint8_t> expectedReadPtr{0x34, 0x12, 0x88};
EXPECT_CALL(*dataInterfaceMockPtr, write(expectedBmcReadPtrOffset,
ElementsAreArray(expectedReadPtr)))
.WillOnce(Return(expectedWriteSize));
EXPECT_NO_THROW(bufferImpl->updateReadPtr(testNewReadPtr));
auto cachedHeader = bufferImpl->getCachedBufferHeader();
EXPECT_EQ(boost::endian::little_to_native(cachedHeader.bmcReadPtr),
0x881234);
}
TEST_F(BufferTest, BufferUpdateBmcFlagsFail)
{
// Return write size that is not 4 which is sizeof(little_uint32_t)
constexpr size_t wrongWriteSize = 1;
EXPECT_CALL(*dataInterfaceMockPtr, write(_, _))
.WillOnce(Return(wrongWriteSize));
EXPECT_THROW(
try {
bufferImpl->updateBmcFlags(static_cast<uint32_t>(BmcFlags::ready));
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[updateBmcFlags] Wrote '1' bytes, instead of expected '4'");
throw;
},
std::runtime_error);
}
TEST_F(BufferTest, BufferUpdateBmcFlagsPass)
{
constexpr size_t expectedWriteSize = 4;
constexpr uint8_t expectedBmcReadPtrOffset = 0x1d;
const std::vector<uint8_t> expectedNewBmcFlagsVector{0x04, 0x0, 0x0, 0x00};
EXPECT_CALL(*dataInterfaceMockPtr,
write(expectedBmcReadPtrOffset,
ElementsAreArray(expectedNewBmcFlagsVector)))
.WillOnce(Return(expectedWriteSize));
EXPECT_NO_THROW(
bufferImpl->updateBmcFlags(static_cast<uint32_t>(BmcFlags::ready)));
auto cachedHeader = bufferImpl->getCachedBufferHeader();
EXPECT_EQ(boost::endian::little_to_native(cachedHeader.bmcFlags),
static_cast<uint32_t>(BmcFlags::ready));
}
TEST_F(BufferTest, GetMaxOffsetQueueSizeFail)
{
InSequence s;
static constexpr size_t wrongQueueSize = testQueueSize - 1;
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
const std::vector<uint8_t> emptyArray(wrongQueueSize, 0);
EXPECT_CALL(*dataInterfaceMockPtr, write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(wrongQueueSize));
EXPECT_CALL(*dataInterfaceMockPtr, write(0, _))
.WillOnce(Return(bufferHeaderSize));
EXPECT_NO_THROW(
bufferImpl->initialize(testBmcInterfaceVersion, wrongQueueSize,
testUeRegionSize, testMagicNumber));
EXPECT_THROW(
try {
bufferImpl->getMaxOffset();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(),
"[getMaxOffset] runtime queueSize '511' did not match "
"compile-time queueSize '512'. This indicates that the"
" buffer was corrupted");
throw;
},
std::runtime_error);
}
TEST_F(BufferTest, GetMaxOffsetUeRegionSizeFail)
{
InSequence s;
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
const std::vector<uint8_t> emptyArray(testQueueSize, 0);
EXPECT_CALL(*dataInterfaceMockPtr, write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(testQueueSize));
EXPECT_CALL(*dataInterfaceMockPtr, write(0, _))
.WillOnce(Return(bufferHeaderSize));
EXPECT_NO_THROW(
bufferImpl->initialize(testBmcInterfaceVersion, testQueueSize,
testUeRegionSize + 1, testMagicNumber));
EXPECT_THROW(
try {
bufferImpl->getMaxOffset();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[getMaxOffset] runtime ueRegionSize '81' did not match "
"compile-time ueRegionSize '80'. This indicates that the"
" buffer was corrupted");
throw;
},
std::runtime_error);
}
TEST_F(BufferTest, GetOffsetUeRegionSizeFail)
{
InSequence s;
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
const std::vector<uint8_t> emptyArray(testQueueSize, 0);
EXPECT_CALL(*dataInterfaceMockPtr, write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(testQueueSize));
EXPECT_CALL(*dataInterfaceMockPtr, write(0, _))
.WillOnce(Return(bufferHeaderSize));
EXPECT_NO_THROW(
bufferImpl->initialize(testBmcInterfaceVersion, testQueueSize,
testUeRegionSize - 1, testMagicNumber));
EXPECT_THROW(
try {
bufferImpl->getQueueOffset();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[getQueueOffset] runtime ueRegionSize '79' did not match "
"compile-time ueRegionSize '80'. This indicates that the"
" buffer was corrupted");
throw;
},
std::runtime_error);
}
class BufferWraparoundReadTest : public BufferTest
{
protected:
BufferWraparoundReadTest()
{
initializeFuncMock();
}
void initializeFuncMock()
{
// Initialize the memory and the cachedBufferHeader
InSequence s;
EXPECT_CALL(*dataInterfaceMockPtr, getMemoryRegionSize())
.WillOnce(Return(testRegionSize));
const std::vector<uint8_t> emptyArray(testQueueSize, 0);
EXPECT_CALL(*dataInterfaceMockPtr,
write(0, ElementsAreArray(emptyArray)))
.WillOnce(Return(testQueueSize));
EXPECT_CALL(*dataInterfaceMockPtr, write(0, _))
.WillOnce(Return(bufferHeaderSize));
EXPECT_NO_THROW(
bufferImpl->initialize(testBmcInterfaceVersion, testQueueSize,
testUeRegionSize, testMagicNumber));
}
static constexpr size_t expectedWriteSize = 3;
static constexpr uint8_t expectedBmcReadPtrOffset = 0x21;
static constexpr size_t expectedqueueOffset = 0x30 + testUeRegionSize;
static constexpr size_t testMaxOffset =
testQueueSize - testUeRegionSize - sizeof(struct CircularBufferHeader);
uint8_t* testInitializationHeaderPtr =
reinterpret_cast<uint8_t*>(&testInitializationHeader);
};
TEST_F(BufferWraparoundReadTest, GetMaxOffsetPassTest)
{
EXPECT_EQ(bufferImpl->getMaxOffset(), testMaxOffset);
}
TEST_F(BufferWraparoundReadTest, GetQueueOffsetPassTest)
{
EXPECT_EQ(bufferImpl->getQueueOffset(),
bufferHeaderSize + testUeRegionSize);
}
TEST_F(BufferWraparoundReadTest, ParamsTooBigFail)
{
InSequence s;
size_t tooBigOffset = testMaxOffset + 1;
EXPECT_THROW(
try {
bufferImpl->wraparoundRead(tooBigOffset, /* length */ 1);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[wraparoundRead] relativeOffset '385' was bigger than maxOffset '384'");
throw;
},
std::runtime_error);
size_t tooBigLength = testMaxOffset + 1;
EXPECT_THROW(
try {
bufferImpl->wraparoundRead(/* relativeOffset */ 0, tooBigLength);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(), "[wraparoundRead] length '385' was bigger "
"than maxOffset '384'");
throw;
},
std::runtime_error);
}
TEST_F(BufferWraparoundReadTest, NoWrapAroundReadFails)
{
InSequence s;
size_t testLength = 0x10;
size_t testOffset = 0x20;
// Fail the first read
std::vector<std::uint8_t> shortTestBytesRead(testLength - 1);
EXPECT_CALL(*dataInterfaceMockPtr,
read(testOffset + expectedqueueOffset, testLength))
.WillOnce(Return(shortTestBytesRead));
EXPECT_THROW(
try {
bufferImpl->wraparoundRead(testOffset, testLength);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(),
"[wraparoundRead] Read '15' which was not the "
"requested length of '16'");
throw;
},
std::runtime_error);
}
TEST_F(BufferWraparoundReadTest, NoWrapAroundReadPass)
{
InSequence s;
size_t testLength = 0x10;
size_t testOffset = 0x20;
// Successfully read all the requested length without a wrap around
std::vector<std::uint8_t> testBytesRead(testLength);
EXPECT_CALL(*dataInterfaceMockPtr,
read(testOffset + expectedqueueOffset, testLength))
.WillOnce(Return(testBytesRead));
// Call to updateReadPtr is triggered
const std::vector<uint8_t> expectedReadPtr{
static_cast<uint8_t>(testOffset + testLength), 0x0, 0x0};
EXPECT_CALL(*dataInterfaceMockPtr, write(expectedBmcReadPtrOffset,
ElementsAreArray(expectedReadPtr)))
.WillOnce(Return(expectedWriteSize));
EXPECT_THAT(bufferImpl->wraparoundRead(testOffset, testLength),
ElementsAreArray(testBytesRead));
struct CircularBufferHeader cachedBufferHeader =
bufferImpl->getCachedBufferHeader();
// The bmcReadPtr should have been updated
EXPECT_EQ(boost::endian::little_to_native(cachedBufferHeader.bmcReadPtr),
testOffset + testLength);
}
TEST_F(BufferWraparoundReadTest, WrapAroundReadFails)
{
InSequence s;
size_t testBytesLeft = 3;
size_t testLength = 0x10;
size_t testOffset = testMaxOffset - (testLength - testBytesLeft);
// Read until the end of the queue
std::vector<std::uint8_t> testBytesReadShort(testLength - testBytesLeft);
EXPECT_CALL(*dataInterfaceMockPtr, read(testOffset + expectedqueueOffset,
testLength - testBytesLeft))
.WillOnce(Return(testBytesReadShort));
// Read 1 byte short after wraparound
std::vector<std::uint8_t> testBytesLeftReadShort(testBytesLeft - 1);
EXPECT_CALL(*dataInterfaceMockPtr, read(expectedqueueOffset, testBytesLeft))
.WillOnce(Return(testBytesLeftReadShort));
EXPECT_THROW(
try {
bufferImpl->wraparoundRead(testOffset, testLength);
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[wraparoundRead] Buffer wrapped around but read '2' which was "
"not the requested lenght of '3'");
throw;
},
std::runtime_error);
}
TEST_F(BufferWraparoundReadTest, WrapAroundReadPasses)
{
InSequence s;
size_t testBytesLeft = 3;
size_t testLength = 0x10;
size_t testOffset = testMaxOffset - (testLength - testBytesLeft);
// Read to the end of the queue
std::vector<std::uint8_t> testBytesReadFirst{16, 15, 14, 13, 12, 11, 10,
9, 8, 7, 6, 5, 4};
EXPECT_CALL(*dataInterfaceMockPtr, read(testOffset + expectedqueueOffset,
testLength - testBytesLeft))
.WillOnce(Return(testBytesReadFirst));
std::vector<std::uint8_t> testBytesReadSecond{3, 2, 1};
EXPECT_CALL(*dataInterfaceMockPtr, read(expectedqueueOffset, testBytesLeft))
.WillOnce(Return(testBytesReadSecond));
// Call to updateReadPtr is triggered
const std::vector<uint8_t> expectedReadPtr{
static_cast<uint8_t>(testBytesLeft), 0x0, 0x0};
EXPECT_CALL(*dataInterfaceMockPtr, write(expectedBmcReadPtrOffset,
ElementsAreArray(expectedReadPtr)))
.WillOnce(Return(expectedWriteSize));
std::vector<std::uint8_t> expectedBytes = {16, 15, 14, 13, 12, 11, 10, 9,
8, 7, 6, 5, 4, 3, 2, 1};
EXPECT_THAT(bufferImpl->wraparoundRead(testOffset, testLength),
ElementsAreArray(expectedBytes));
struct CircularBufferHeader cachedBufferHeader =
bufferImpl->getCachedBufferHeader();
// The bmcReadPtr should have been updated to reflect the wraparound
EXPECT_EQ(boost::endian::little_to_native(cachedBufferHeader.bmcReadPtr),
testBytesLeft);
}
TEST_F(BufferWraparoundReadTest, WrapAroundCornerCasePass)
{
InSequence s;
size_t testBytesLeft = 0;
size_t testLength = 4;
size_t testOffset = testMaxOffset - (testLength - testBytesLeft);
// Read to the very end of the queue
std::vector<std::uint8_t> testBytes{4, 3, 2, 1};
EXPECT_CALL(*dataInterfaceMockPtr,
read(testOffset + expectedqueueOffset, testLength))
.WillOnce(Return(testBytes));
// Call to updateReadPtr is triggered, since we read to the very end of the
// buffer, update the readPtr up around to 0
const std::vector<uint8_t> expectedReadPtr{0x0, 0x0, 0x0};
EXPECT_CALL(*dataInterfaceMockPtr, write(expectedBmcReadPtrOffset,
ElementsAreArray(expectedReadPtr)))
.WillOnce(Return(expectedWriteSize));
EXPECT_THAT(bufferImpl->wraparoundRead(testOffset, testLength),
ElementsAreArray(testBytes));
struct CircularBufferHeader cachedBufferHeader =
bufferImpl->getCachedBufferHeader();
// The bmcReadPtr should have been updated to reflect the wraparound
EXPECT_EQ(boost::endian::little_to_native(cachedBufferHeader.bmcReadPtr),
0);
}
class BufferEntryTest : public BufferWraparoundReadTest
{
protected:
BufferEntryTest()
{
testEntryHeader.sequenceId = testSequenceId;
testEntryHeader.entrySize = testEntrySize;
testEntryHeader.checksum = testChecksum;
testEntryHeader.rdeCommandType = testRdeCommandType;
}
~BufferEntryTest() override = default;
void wraparoundReadMock(const uint32_t relativeOffset,
std::span<std::uint8_t> expetedBytesOutput)
{
InSequence s;
const uint32_t queueSizeToQueueEnd = testMaxOffset - relativeOffset;
// This will wrap, split the read mocks in 2
if (expetedBytesOutput.size() > queueSizeToQueueEnd)
{
EXPECT_CALL(*dataInterfaceMockPtr, read(_, _))
.WillOnce(Return(std::vector<std::uint8_t>(
expetedBytesOutput.begin(),
expetedBytesOutput.begin() + queueSizeToQueueEnd)));
EXPECT_CALL(*dataInterfaceMockPtr, read(_, _))
.WillOnce(Return(std::vector<std::uint8_t>(
expetedBytesOutput.begin() + queueSizeToQueueEnd,
expetedBytesOutput.end())));
}
else
{
EXPECT_CALL(*dataInterfaceMockPtr, read(_, _))
.WillOnce(Return(std::vector<std::uint8_t>(
expetedBytesOutput.begin(), expetedBytesOutput.end())));
}
EXPECT_CALL(*dataInterfaceMockPtr, write(_, _))
.WillOnce(Return(expectedWriteSize));
}
static constexpr size_t entryHeaderSize = sizeof(struct QueueEntryHeader);
static constexpr uint16_t testSequenceId = 0;
static constexpr uint16_t testEntrySize = 0x20;
static constexpr uint8_t testRdeCommandType = 0x01;
// Calculated checksum for the header
static constexpr uint8_t testChecksum =
(testSequenceId ^ testEntrySize ^ testRdeCommandType);
size_t testOffset = 0x0;
struct QueueEntryHeader testEntryHeader{};
};
TEST_F(BufferEntryTest, ReadEntryHeaderPass)
{
uint8_t* testEntryHeaderPtr = reinterpret_cast<uint8_t*>(&testEntryHeader);
std::vector<uint8_t> testEntryHeaderVector(
testEntryHeaderPtr, testEntryHeaderPtr + entryHeaderSize);
wraparoundReadMock(testOffset, testEntryHeaderVector);
EXPECT_EQ(bufferImpl->readEntryHeader(), testEntryHeader);
// Check the bmcReadPtr
struct CircularBufferHeader cachedBufferHeader =
bufferImpl->getCachedBufferHeader();
EXPECT_EQ(boost::endian::little_to_native(cachedBufferHeader.bmcReadPtr),
testOffset + testEntryHeaderVector.size());
}
TEST_F(BufferEntryTest, ReadEntryChecksumFail)
{
InSequence s;
std::vector<uint8_t> testEntryVector(testEntrySize);
// Offset the checksum by 1
testEntryHeader.checksum += 1;
uint8_t* testEntryHeaderPtr = reinterpret_cast<uint8_t*>(&testEntryHeader);
std::vector<uint8_t> testEntryHeaderVector(
testEntryHeaderPtr, testEntryHeaderPtr + entryHeaderSize);
wraparoundReadMock(testOffset, testEntryHeaderVector);
wraparoundReadMock(testOffset + entryHeaderSize, testEntryVector);
EXPECT_THROW(
try { bufferImpl->readEntry(); } catch (const std::runtime_error& e) {
// Calculation: testChecksum (0x21) XOR (0x22) = 3
EXPECT_STREQ(e.what(),
"[readEntry] Checksum was '3', expected '0'");
throw;
},
std::runtime_error);
}
TEST_F(BufferEntryTest, ReadEntryPassWraparound)
{
InSequence s;
// We expect this will bump checksum up by "testEntrySize" = 0xff ^ 0xff ...
// (20 times) = 0 therefore leave the checksum as is
std::vector<uint8_t> testEntryVector(testEntrySize, 0xff);
uint8_t* testEntryHeaderPtr = reinterpret_cast<uint8_t*>(&testEntryHeader);
std::vector<uint8_t> testEntryHeaderVector(
testEntryHeaderPtr, testEntryHeaderPtr + entryHeaderSize);
// Set testOffset so that we can test the wraparound here at the header and
// update the readPtr
testOffset = testMaxOffset - 1;
EXPECT_CALL(*dataInterfaceMockPtr, write(expectedBmcReadPtrOffset, _))
.WillOnce(Return(expectedWriteSize));
EXPECT_NO_THROW(bufferImpl->updateReadPtr(testOffset));
wraparoundReadMock(testOffset, testEntryHeaderVector);
wraparoundReadMock(testOffset + entryHeaderSize, testEntryVector);
EntryPair testedEntryPair;
EXPECT_NO_THROW(testedEntryPair = bufferImpl->readEntry());
EXPECT_EQ(testedEntryPair.first, testEntryHeader);
EXPECT_THAT(testedEntryPair.second, ElementsAreArray(testEntryVector));
struct CircularBufferHeader cachedBufferHeader =
bufferImpl->getCachedBufferHeader();
// The bmcReadPtr should have been updated to reflect the wraparound
EXPECT_EQ(boost::endian::little_to_native(cachedBufferHeader.bmcReadPtr),
entryHeaderSize + testEntrySize - 1);
// Set testOffset so that we can test the wraparound here as well on our
// second read for the entry (by 1 byte)
testOffset = testMaxOffset - entryHeaderSize - 1;
EXPECT_CALL(*dataInterfaceMockPtr, write(expectedBmcReadPtrOffset, _))
.WillOnce(Return(expectedWriteSize));
EXPECT_NO_THROW(bufferImpl->updateReadPtr(testOffset));
wraparoundReadMock(testOffset, testEntryHeaderVector);
wraparoundReadMock(testOffset + entryHeaderSize, testEntryVector);
EXPECT_NO_THROW(testedEntryPair = bufferImpl->readEntry());
EXPECT_EQ(testedEntryPair.first, testEntryHeader);
EXPECT_THAT(testedEntryPair.second, ElementsAreArray(testEntryVector));
cachedBufferHeader = bufferImpl->getCachedBufferHeader();
// The bmcReadPtr should have been updated to reflect the wraparound
EXPECT_EQ(boost::endian::little_to_native(cachedBufferHeader.bmcReadPtr),
testEntrySize - 1);
}
class BufferReadErrorLogsTest : public BufferEntryTest
{
protected:
BufferReadErrorLogsTest() = default;
uint8_t* testEntryHeaderPtr = reinterpret_cast<uint8_t*>(&testEntryHeader);
size_t entryAndHeaderSize = entryHeaderSize + testEntrySize;
};
TEST_F(BufferReadErrorLogsTest, PtrsTooBigFail)
{
InSequence s;
// Set the biosWritePtr too big
testInitializationHeader.biosWritePtr =
boost::endian::native_to_little((testMaxOffset + 1));
initializeFuncMock();
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(std::vector<uint8_t>(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize)));
EXPECT_THROW(
try {
bufferImpl->readErrorLogs();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(),
"[readErrorLogs] currentBiosWritePtr was '385' "
"which was bigger than maxOffset '384'");
throw;
},
std::runtime_error);
// Reset the biosWritePtr and set the bmcReadPtr too big
testInitializationHeader.biosWritePtr = 0;
initializeFuncMock();
testInitializationHeader.bmcReadPtr =
boost::endian::native_to_little((testMaxOffset + 1));
initializeFuncMock();
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(std::vector<uint8_t>(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize)));
EXPECT_THROW(
try {
bufferImpl->readErrorLogs();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(e.what(), "[readErrorLogs] currentReadPtr was '385' "
"which was bigger than maxOffset '384'");
throw;
},
std::runtime_error);
}
TEST_F(BufferReadErrorLogsTest, IdenticalPtrsPass)
{
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(std::vector<uint8_t>(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize)));
EXPECT_NO_THROW(bufferImpl->readErrorLogs());
}
TEST_F(BufferReadErrorLogsTest, NoWraparoundPass)
{
InSequence s;
// Set the biosWritePtr to 1 entryHeader + entry size
testInitializationHeader.biosWritePtr =
boost::endian::native_to_little((entryAndHeaderSize));
initializeFuncMock();
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(std::vector<uint8_t>(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize)));
std::vector<uint8_t> testEntryHeaderVector(
testEntryHeaderPtr, testEntryHeaderPtr + entryHeaderSize);
std::vector<uint8_t> testEntryVector(testEntrySize);
wraparoundReadMock(/*relativeOffset=*/0, testEntryHeaderVector);
wraparoundReadMock(/*relativeOffset=*/0 + entryHeaderSize, testEntryVector);
std::vector<EntryPair> entryPairs;
EXPECT_NO_THROW(entryPairs = bufferImpl->readErrorLogs());
// Check that we only read one entryPair and that the content is correct
EXPECT_EQ(entryPairs.size(), 1U);
EXPECT_EQ(entryPairs[0].first, testEntryHeader);
EXPECT_THAT(entryPairs[0].second, ElementsAreArray(testEntryVector));
}
TEST_F(BufferReadErrorLogsTest, WraparoundMultiplEntryPass)
{
InSequence s;
// Set the bmcReadPtr to 1 entryHeader + entry size from the "end" exactly
uint32_t entryAndHeaderSizeAwayFromEnd = testMaxOffset - entryAndHeaderSize;
testInitializationHeader.bmcReadPtr =
boost::endian::native_to_little(entryAndHeaderSizeAwayFromEnd);
// Set the biosWritePtr to 1 entryHeader + entry size from the "beginning"
testInitializationHeader.biosWritePtr = entryAndHeaderSize;
initializeFuncMock();
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(std::vector<uint8_t>(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize)));
std::vector<uint8_t> testEntryHeaderVector(
testEntryHeaderPtr, testEntryHeaderPtr + entryHeaderSize);
std::vector<uint8_t> testEntryVector(testEntrySize);
wraparoundReadMock(/*relativeOffset=*/entryAndHeaderSizeAwayFromEnd,
testEntryHeaderVector);
wraparoundReadMock(/*relativeOffset=*/entryAndHeaderSizeAwayFromEnd +
entryHeaderSize,
testEntryVector);
wraparoundReadMock(/*relativeOffset=*/0 + entryAndHeaderSize,
testEntryHeaderVector);
wraparoundReadMock(/*relativeOffset=*/0 + entryAndHeaderSize +
entryHeaderSize,
testEntryVector);
std::vector<EntryPair> entryPairs;
EXPECT_NO_THROW(entryPairs = bufferImpl->readErrorLogs());
// Check that we only read one entryPair and that the content is correct
EXPECT_EQ(entryPairs.size(), 2);
EXPECT_EQ(entryPairs[0].first, testEntryHeader);
EXPECT_EQ(entryPairs[1].first, testEntryHeader);
EXPECT_THAT(entryPairs[0].second, ElementsAreArray(testEntryVector));
EXPECT_THAT(entryPairs[1].second, ElementsAreArray(testEntryVector));
}
TEST_F(BufferReadErrorLogsTest, WraparoundMismatchingPtrsFail)
{
InSequence s;
testInitializationHeader.bmcReadPtr = boost::endian::native_to_little(0);
// Make the biosWritePtr intentially 1 smaller than expected
testInitializationHeader.biosWritePtr =
boost::endian::native_to_little(entryAndHeaderSize - 1);
initializeFuncMock();
EXPECT_CALL(*dataInterfaceMockPtr, read(0, bufferHeaderSize))
.WillOnce(Return(std::vector<uint8_t>(
testInitializationHeaderPtr,
testInitializationHeaderPtr + bufferHeaderSize)));
std::vector<uint8_t> testEntryHeaderVector(
testEntryHeaderPtr, testEntryHeaderPtr + entryHeaderSize);
std::vector<uint8_t> testEntryVector(testEntrySize);
wraparoundReadMock(/*relativeOffset=*/0, testEntryHeaderVector);
wraparoundReadMock(/*relativeOffset=*/0 + entryHeaderSize, testEntryVector);
EXPECT_THROW(
try {
bufferImpl->readErrorLogs();
} catch (const std::runtime_error& e) {
EXPECT_STREQ(
e.what(),
"[readErrorLogs] biosWritePtr '37' and bmcReaddPtr '38' "
"are not identical after reading through all the logs");
throw;
},
std::runtime_error);
}
} // namespace
} // namespace bios_bmc_smm_error_logger