1e705b37aSFangrui Song #include "llvm/Transforms/Utils/CodeLayout.h"
2e705b37aSFangrui Song #include "gmock/gmock.h"
3e705b37aSFangrui Song #include "gtest/gtest.h"
4e705b37aSFangrui Song #include <vector>
5e705b37aSFangrui Song
6e705b37aSFangrui Song using namespace llvm;
7e705b37aSFangrui Song using namespace llvm::codelayout;
8e705b37aSFangrui Song using testing::ElementsAreArray;
9e705b37aSFangrui Song
10e705b37aSFangrui Song namespace {
TEST(CodeLayout,ThreeFunctions)11e705b37aSFangrui Song TEST(CodeLayout, ThreeFunctions) {
12e705b37aSFangrui Song // Place the most likely successor (2) first.
13e705b37aSFangrui Song {
14e705b37aSFangrui Song const uint64_t Counts[3] = {140, 40, 140};
15e705b37aSFangrui Song const std::vector<uint64_t> Sizes(std::size(Counts), 9);
16e705b37aSFangrui Song const EdgeCount Edges[] = {{0, 1, 40}, {0, 2, 100}, {1, 2, 40}};
17e705b37aSFangrui Song const std::vector<uint64_t> CallOffsets(std::size(Edges), 5);
18e705b37aSFangrui Song auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
19e705b37aSFangrui Song EXPECT_THAT(Order, ElementsAreArray({0, 2, 1}));
20e705b37aSFangrui Song }
21e705b37aSFangrui Song
22e705b37aSFangrui Song // Prefer fallthroughs even in the presence of a heavy successor.
23e705b37aSFangrui Song {
24e705b37aSFangrui Song const uint64_t Counts[3] = {180, 80, 180};
25e705b37aSFangrui Song const std::vector<uint64_t> Sizes(std::size(Counts), 9);
26e705b37aSFangrui Song const EdgeCount Edges[] = {{0, 1, 80}, {0, 2, 100}, {1, 2, 80}};
27e705b37aSFangrui Song const uint64_t CallOffsets[] = {9, 5, 9};
28e705b37aSFangrui Song auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
29e705b37aSFangrui Song EXPECT_THAT(Order, ElementsAreArray({0, 1, 2}));
30e705b37aSFangrui Song }
31e705b37aSFangrui Song }
32e705b37aSFangrui Song
TEST(CodeLayout,HotChain)33e705b37aSFangrui Song TEST(CodeLayout, HotChain) {
34e705b37aSFangrui Song // Place the hot chain (0,3,4,2) continuously.
35e705b37aSFangrui Song {
36e705b37aSFangrui Song const uint64_t Counts[5] = {22, 7, 22, 15, 46};
37e705b37aSFangrui Song const std::vector<uint64_t> Sizes(std::size(Counts), 9);
38e705b37aSFangrui Song const EdgeCount Edges[] = {{0, 1, 7}, {1, 2, 7}, {0, 3, 15},
39e705b37aSFangrui Song {3, 4, 15}, {4, 4, 31}, {4, 2, 15}};
40e705b37aSFangrui Song const std::vector<uint64_t> CallOffsets(std::size(Edges), 5);
41e705b37aSFangrui Song auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
42e705b37aSFangrui Song EXPECT_THAT(Order, ElementsAreArray({0, 3, 4, 2, 1}));
43*a2441837SFangrui Song
44*a2441837SFangrui Song // -cdsort-max-chain-size disables forming a larger chain and therefore may
45*a2441837SFangrui Song // change the result.
46*a2441837SFangrui Song CDSortConfig Config;
47*a2441837SFangrui Song Config.MaxChainSize = 3;
48*a2441837SFangrui Song Order =
49*a2441837SFangrui Song computeCacheDirectedLayout(Config, Sizes, Counts, Edges, CallOffsets);
50*a2441837SFangrui Song EXPECT_THAT(Order, ElementsAreArray({0, 3, 4, 1, 2}));
51e705b37aSFangrui Song }
52e705b37aSFangrui Song }
53e705b37aSFangrui Song
TEST(CodeLayout,BreakLoop)54e705b37aSFangrui Song TEST(CodeLayout, BreakLoop) {
55e705b37aSFangrui Song // There are two loops (1,2,3) and (1,2,4). It is beneficial to place 4
56e705b37aSFangrui Song // elsewhere.
57e705b37aSFangrui Song const uint64_t Counts[6] = {177, 371, 196, 124, 70, 177};
58e705b37aSFangrui Song std::vector<uint64_t> Sizes(std::size(Counts), 9);
59e705b37aSFangrui Song const EdgeCount Edges[] = {{0, 1, 177}, {1, 2, 196}, {2, 3, 124}, {3, 1, 124},
60e705b37aSFangrui Song {1, 5, 177}, {2, 4, 79}, {4, 1, 70}};
61e705b37aSFangrui Song const std::vector<uint64_t> CallOffsets(std::size(Edges), 5);
62e705b37aSFangrui Song auto Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
63e705b37aSFangrui Song EXPECT_THAT(Order, ElementsAreArray({4, 0, 1, 2, 3, 5}));
64e705b37aSFangrui Song
65e705b37aSFangrui Song // When node 0 is larger, it is beneficial to move node 4 closer to the
66e705b37aSFangrui Song // (1,2,3) loop.
67e705b37aSFangrui Song Sizes[0] = 18;
68e705b37aSFangrui Song Order = computeCacheDirectedLayout(Sizes, Counts, Edges, CallOffsets);
69e705b37aSFangrui Song EXPECT_THAT(Order, ElementsAreArray({0, 4, 1, 2, 3, 5}));
70e705b37aSFangrui Song }
71e705b37aSFangrui Song } // namespace
72