aboutsummaryrefslogtreecommitdiff
path: root/llvm/unittests/Transforms/Utils/CodeLayoutTest.cpp
blob: ef9aa9a76342f4d0467f3ff2c17645a05a6ba764 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
#include "llvm/Transforms/Utils/CodeLayout.h"
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include <vector>

using namespace llvm;
using namespace llvm::codelayout;
using testing::ElementsAreArray;

namespace {
TEST(CodeLayout, ThreeFunctions) {
  // Place the most likely successor (2) first.
  {
    const uint64_t Counts[3] = {140, 40, 140};
    const std::vector<uint64_t> Sizes(std::size(Counts), 9);
    const EdgeCount Edges[] = {{0, 1, 40}, {0, 2, 100}, {1, 2, 40}};
    const std::vector<uint64_t> CallOffsets(std::size(Edges), 5);
    auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
    EXPECT_THAT(Order, ElementsAreArray({0, 2, 1}));
  }

  // Prefer fallthroughs even in the presence of a heavy successor.
  {
    const uint64_t Counts[3] = {180, 80, 180};
    const std::vector<uint64_t> Sizes(std::size(Counts), 9);
    const EdgeCount Edges[] = {{0, 1, 80}, {0, 2, 100}, {1, 2, 80}};
    const uint64_t CallOffsets[] = {9, 5, 9};
    auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
    EXPECT_THAT(Order, ElementsAreArray({0, 1, 2}));
  }
}

TEST(CodeLayout, HotChain) {
  // Place the hot chain (0,3,4,2) continuously.
  {
    const uint64_t Counts[5] = {22, 7, 22, 15, 46};
    const std::vector<uint64_t> Sizes(std::size(Counts), 9);
    const EdgeCount Edges[] = {{0, 1, 7},  {1, 2, 7},  {0, 3, 15},
                               {3, 4, 15}, {4, 4, 31}, {4, 2, 15}};
    const std::vector<uint64_t> CallOffsets(std::size(Edges), 5);
    auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
    EXPECT_THAT(Order, ElementsAreArray({0, 3, 4, 2, 1}));

    // -cdsort-max-chain-size disables forming a larger chain and therefore may
    // change the result.
    CDSortConfig Config;
    Config.MaxChainSize = 3;
    Order =
        computeCacheDirectedLayout(Config, Sizes, Counts, Edges, CallOffsets);
    EXPECT_THAT(Order, ElementsAreArray({0, 3, 4, 1, 2}));
  }
}

TEST(CodeLayout, BreakLoop) {
  // There are two loops (1,2,3) and (1,2,4). It is beneficial to place 4
  // elsewhere.
  const uint64_t Counts[6] = {177, 371, 196, 124, 70, 177};
  std::vector<uint64_t> Sizes(std::size(Counts), 9);
  const EdgeCount Edges[] = {{0, 1, 177}, {1, 2, 196}, {2, 3, 124}, {3, 1, 124},
                             {1, 5, 177}, {2, 4, 79},  {4, 1, 70}};
  const std::vector<uint64_t> CallOffsets(std::size(Edges), 5);
  auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
  EXPECT_THAT(Order, ElementsAreArray({4, 0, 1, 2, 3, 5}));

  // When node 0 is larger, it is beneficial to move node 4 closer to the
  // (1,2,3) loop.
  Sizes[0] = 18;
  Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
  EXPECT_THAT(Order, ElementsAreArray({0, 4, 1, 2, 3, 5}));
}
} // namespace