diff options
Diffstat (limited to 'llvm/unittests/Transforms/IPO/WholeProgramDevirt.cpp')
-rw-r--r-- | llvm/unittests/Transforms/IPO/WholeProgramDevirt.cpp | 164 |
1 files changed, 164 insertions, 0 deletions
diff --git a/llvm/unittests/Transforms/IPO/WholeProgramDevirt.cpp b/llvm/unittests/Transforms/IPO/WholeProgramDevirt.cpp new file mode 100644 index 00000000000..2a79f02f83d --- /dev/null +++ b/llvm/unittests/Transforms/IPO/WholeProgramDevirt.cpp @@ -0,0 +1,164 @@ +//===- WholeProgramDevirt.cpp - Unit tests for whole-program devirt -------===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "llvm/Transforms/IPO/WholeProgramDevirt.h" +#include "gtest/gtest.h" + +using namespace llvm; +using namespace wholeprogramdevirt; + +TEST(WholeProgramDevirt, findLowestOffset) { + VTableBits VT1; + VT1.ObjectSize = 8; + VT1.Before.BytesUsed = {1 << 0}; + VT1.After.BytesUsed = {1 << 1}; + + VTableBits VT2; + VT2.ObjectSize = 8; + VT2.Before.BytesUsed = {1 << 1}; + VT2.After.BytesUsed = {1 << 0}; + + BitSetInfo BS1{&VT1, 0}; + BitSetInfo BS2{&VT2, 0}; + VirtualCallTarget Targets[] = { + {&BS1, /*IsBigEndian=*/false}, + {&BS2, /*IsBigEndian=*/false}, + }; + + EXPECT_EQ(2ull, findLowestOffset(Targets, /*IsAfter=*/false, 1)); + EXPECT_EQ(66ull, findLowestOffset(Targets, /*IsAfter=*/true, 1)); + + EXPECT_EQ(8ull, findLowestOffset(Targets, /*IsAfter=*/false, 8)); + EXPECT_EQ(72ull, findLowestOffset(Targets, /*IsAfter=*/true, 8)); + + BS1.Offset = 4; + EXPECT_EQ(33ull, findLowestOffset(Targets, /*IsAfter=*/false, 1)); + EXPECT_EQ(65ull, findLowestOffset(Targets, /*IsAfter=*/true, 1)); + + EXPECT_EQ(40ull, findLowestOffset(Targets, /*IsAfter=*/false, 8)); + EXPECT_EQ(72ull, findLowestOffset(Targets, /*IsAfter=*/true, 8)); + + BS1.Offset = 8; + BS2.Offset = 8; + EXPECT_EQ(66ull, findLowestOffset(Targets, /*IsAfter=*/false, 1)); + EXPECT_EQ(2ull, findLowestOffset(Targets, /*IsAfter=*/true, 1)); + + EXPECT_EQ(72ull, findLowestOffset(Targets, /*IsAfter=*/false, 8)); + EXPECT_EQ(8ull, findLowestOffset(Targets, /*IsAfter=*/true, 8)); + + VT1.After.BytesUsed = {0xff, 0, 0, 0, 0xff}; + VT2.After.BytesUsed = {0xff, 1, 0, 0, 0}; + EXPECT_EQ(16ull, findLowestOffset(Targets, /*IsAfter=*/true, 16)); + EXPECT_EQ(40ull, findLowestOffset(Targets, /*IsAfter=*/true, 32)); +} + +TEST(WholeProgramDevirt, setReturnValues) { + VTableBits VT1; + VT1.ObjectSize = 8; + + VTableBits VT2; + VT2.ObjectSize = 8; + + BitSetInfo BS1{&VT1, 0}; + BitSetInfo BS2{&VT2, 0}; + VirtualCallTarget Targets[] = { + {&BS1, /*IsBigEndian=*/false}, + {&BS2, /*IsBigEndian=*/false}, + }; + + BS1.Offset = 4; + BS2.Offset = 4; + + int64_t OffsetByte; + uint64_t OffsetBit; + + Targets[0].RetVal = 1; + Targets[1].RetVal = 0; + setBeforeReturnValues(Targets, 32, 1, OffsetByte, OffsetBit); + EXPECT_EQ(-5ll, OffsetByte); + EXPECT_EQ(0ull, OffsetBit); + EXPECT_EQ(std::vector<uint8_t>{1}, VT1.Before.Bytes); + EXPECT_EQ(std::vector<uint8_t>{1}, VT1.Before.BytesUsed); + EXPECT_EQ(std::vector<uint8_t>{0}, VT2.Before.Bytes); + EXPECT_EQ(std::vector<uint8_t>{1}, VT2.Before.BytesUsed); + + Targets[0].RetVal = 0; + Targets[1].RetVal = 1; + setBeforeReturnValues(Targets, 39, 1, OffsetByte, OffsetBit); + EXPECT_EQ(-5ll, OffsetByte); + EXPECT_EQ(7ull, OffsetBit); + EXPECT_EQ(std::vector<uint8_t>{1}, VT1.Before.Bytes); + EXPECT_EQ(std::vector<uint8_t>{0x81}, VT1.Before.BytesUsed); + EXPECT_EQ(std::vector<uint8_t>{0x80}, VT2.Before.Bytes); + EXPECT_EQ(std::vector<uint8_t>{0x81}, VT2.Before.BytesUsed); + + Targets[0].RetVal = 12; + Targets[1].RetVal = 34; + setBeforeReturnValues(Targets, 40, 8, OffsetByte, OffsetBit); + EXPECT_EQ(-6ll, OffsetByte); + EXPECT_EQ(0ull, OffsetBit); + EXPECT_EQ((std::vector<uint8_t>{1, 12}), VT1.Before.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff}), VT1.Before.BytesUsed); + EXPECT_EQ((std::vector<uint8_t>{0x80, 34}), VT2.Before.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff}), VT2.Before.BytesUsed); + + Targets[0].RetVal = 56; + Targets[1].RetVal = 78; + setBeforeReturnValues(Targets, 48, 16, OffsetByte, OffsetBit); + EXPECT_EQ(-8ll, OffsetByte); + EXPECT_EQ(0ull, OffsetBit); + EXPECT_EQ((std::vector<uint8_t>{1, 12, 0, 56}), VT1.Before.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff, 0xff, 0xff}), + VT1.Before.BytesUsed); + EXPECT_EQ((std::vector<uint8_t>{0x80, 34, 0, 78}), VT2.Before.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff, 0xff, 0xff}), + VT2.Before.BytesUsed); + + Targets[0].RetVal = 1; + Targets[1].RetVal = 0; + setAfterReturnValues(Targets, 32, 1, OffsetByte, OffsetBit); + EXPECT_EQ(4ll, OffsetByte); + EXPECT_EQ(0ull, OffsetBit); + EXPECT_EQ(std::vector<uint8_t>{1}, VT1.After.Bytes); + EXPECT_EQ(std::vector<uint8_t>{1}, VT1.After.BytesUsed); + EXPECT_EQ(std::vector<uint8_t>{0}, VT2.After.Bytes); + EXPECT_EQ(std::vector<uint8_t>{1}, VT2.After.BytesUsed); + + Targets[0].RetVal = 0; + Targets[1].RetVal = 1; + setAfterReturnValues(Targets, 39, 1, OffsetByte, OffsetBit); + EXPECT_EQ(4ll, OffsetByte); + EXPECT_EQ(7ull, OffsetBit); + EXPECT_EQ(std::vector<uint8_t>{1}, VT1.After.Bytes); + EXPECT_EQ(std::vector<uint8_t>{0x81}, VT1.After.BytesUsed); + EXPECT_EQ(std::vector<uint8_t>{0x80}, VT2.After.Bytes); + EXPECT_EQ(std::vector<uint8_t>{0x81}, VT2.After.BytesUsed); + + Targets[0].RetVal = 12; + Targets[1].RetVal = 34; + setAfterReturnValues(Targets, 40, 8, OffsetByte, OffsetBit); + EXPECT_EQ(5ll, OffsetByte); + EXPECT_EQ(0ull, OffsetBit); + EXPECT_EQ((std::vector<uint8_t>{1, 12}), VT1.After.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff}), VT1.After.BytesUsed); + EXPECT_EQ((std::vector<uint8_t>{0x80, 34}), VT2.After.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff}), VT2.After.BytesUsed); + + Targets[0].RetVal = 56; + Targets[1].RetVal = 78; + setAfterReturnValues(Targets, 48, 16, OffsetByte, OffsetBit); + EXPECT_EQ(6ll, OffsetByte); + EXPECT_EQ(0ull, OffsetBit); + EXPECT_EQ((std::vector<uint8_t>{1, 12, 56, 0}), VT1.After.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff, 0xff, 0xff}), + VT1.After.BytesUsed); + EXPECT_EQ((std::vector<uint8_t>{0x80, 34, 78, 0}), VT2.After.Bytes); + EXPECT_EQ((std::vector<uint8_t>{0x81, 0xff, 0xff, 0xff}), + VT2.After.BytesUsed); +} |