-
Notifications
You must be signed in to change notification settings - Fork 63
/
loadstore-benches.cpp
175 lines (143 loc) · 6.3 KB
/
loadstore-benches.cpp
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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
/*
* loadstore_benches.cpp
*/
#include <iostream>
#include <cassert>
#include <sstream>
#include <iomanip>
#include "benchmark.hpp"
#include "hedley.h"
#include "util.hpp"
#include "context.hpp"
#include "timers.hpp"
#include "table.hpp"
#include "isa-support.hpp"
extern "C" {
bench2_f store16_any;
bench2_f store32_any;
bench2_f store64_any;
bench2_f store128_any; // AVX2 128-bit store
bench2_f store256_any; // AVX2 256-bit store
bench2_f store512_any; // AVX512F 512-bit store
bench2_f load16_any;
bench2_f load32_any;
bench2_f load64_any;
bench2_f load128_any; // AVX (REX-encoded) 128-bit load
bench2_f load256_any; // AVX (REX-encoded) 256-bit load
bench2_f load512_any; // AVX512F 512-bit load
}
using namespace std;
/*
* A specialization of BenchmarkGroup that outputs its results in a 4 x 16 grid for all 64 possible
* offsets within a 64B cache line.
*/
class LoadStoreGroup : public BenchmarkGroup {
static constexpr unsigned DEFAULT_ROWS = 4;
static constexpr unsigned DEFAULT_COLS = 16;
unsigned rows_, cols_, op_size_;
public:
LoadStoreGroup(const string& id, const string& name, unsigned op_size, unsigned rows, unsigned cols)
: BenchmarkGroup(id, name), rows_(rows), cols_(cols), op_size_(op_size) {
assert(rows < 10000 && cols < 10000);
}
HEDLEY_NEVER_INLINE std::string make_name(ssize_t misalign);
HEDLEY_NEVER_INLINE std::string make_id(ssize_t misalign);
HEDLEY_NEVER_INLINE static shared_ptr<LoadStoreGroup> make_group(const string& id, const string& name, ssize_t op_size);
template<typename TIMER, bench2_f METHOD>
static shared_ptr<LoadStoreGroup> make(const string& id, unsigned op_size, featurelist_t features = {}) {
shared_ptr<LoadStoreGroup> group = make_group(id, id, op_size);
auto maker = DeltaMaker<TIMER>(group.get(), 1000).setFeatures(features);
for (ssize_t misalign = 0; misalign < 64; misalign++) {
maker.template make<METHOD>(group->make_id(misalign), group->make_name(misalign), 128,
[misalign]() { return misaligned_ptr(64, 64, misalign); });
}
return group;
}
virtual void runIf(Context& c, const predicate_t& predicate) override {
auto benches = getBenches();
if (!supports(benches.front()->getFeatures())) {
// skip tests if we don't have the supported features
return;
}
// because this BenchmarkGroup is really more like a single benchmark (i.e., the 64 actual Benchmark objects
// don't their name printed but are show in a grid instead, we run the predicate on a fake Benchmark created
// based on the group name
Benchmark fake = StaticMaker<DefaultClockTimer>::make_bench<dummy_bench>(this, "fake", getDescription(), 1);
if (!predicate(fake)) {
return;
}
std::ostream& os = c.out();
os << endl << "** Inverse throughput for " << getDescription() << " **" << endl;
// column headers
os << "offset ";
for (unsigned col = 0; col < cols_; col++) {
os << setw(5) << col;
}
os << endl;
assert(benches.size() == rows_ * cols_);
// collect all the results up front, before any output
vector<double> results(benches.size());
for (size_t i = 0; i < benches.size(); i++) {
Benchmark& b = benches[i];
results[i] = b->run(c.getTimerInfo()).getCycles();
}
for (unsigned row = 0, i = 0; row < rows_; row++) {
os << setw(3) << (row * cols_) << " : ";
for (unsigned col = 0; col < cols_; col++, i++) {
os << setprecision(1) << fixed << setw(5) << results[i];
}
os << endl;
}
}
void printBenches(std::ostream& out) const override {
using namespace table;
auto& benches = getBenches();
assert(benches.size() >= 2);
Table t;
t.colInfo(0).justify = ColInfo::LEFT;
t.newRow().add("ID").add("Description");
for (const auto& bench : {benches[0], benches[1], benches.back()}) {
t.newRow().add(bench->getPath()).add(bench->getDescription());
if (bench == benches[1]) {
t.newRow().add("...").add(std::to_string(benches.size() - 3) + " more benches not shown");
}
}
out << t.str();
}
};
constexpr unsigned LoadStoreGroup::DEFAULT_ROWS;
constexpr unsigned LoadStoreGroup::DEFAULT_COLS;
shared_ptr<LoadStoreGroup> LoadStoreGroup::make_group(const string& id, const string& name, ssize_t op_size) {
return make_shared<LoadStoreGroup>(id, name, op_size, DEFAULT_ROWS, DEFAULT_COLS);
}
std::string LoadStoreGroup::make_name(ssize_t misalign) {
std::stringstream ss;
ss << "Misaligned " << (op_size_ * 8) << "-bit " << getDescription() << " [" << setw(2) << misalign << "]";
return ss.str();
}
std::string LoadStoreGroup::make_id(ssize_t misalign) {
std::stringstream ss;
ss << "misaligned-" << misalign;
return ss.str();
}
template <typename TIMER>
void register_loadstore(GroupList& list) {
#if !UARCH_BENCH_PORTABLE
// load throughput benches
list.push_back(LoadStoreGroup::make<TIMER, load16_any>("load/16-bit", 2));
list.push_back(LoadStoreGroup::make<TIMER, load32_any>("load/32-bit", 4));
list.push_back(LoadStoreGroup::make<TIMER, load64_any>("load/64-bit", 8));
list.push_back(LoadStoreGroup::make<TIMER, load128_any>("load/128-bit", 16));
list.push_back(LoadStoreGroup::make<TIMER, load256_any>("load/256-bit", 32));
list.push_back(LoadStoreGroup::make<TIMER, load512_any>("load/512-bit", 64, { x86Feature::AVX512F }));
// store throughput
list.push_back(LoadStoreGroup::make<TIMER, store16_any>( "store/16-bit", 2));
list.push_back(LoadStoreGroup::make<TIMER, store32_any>( "store/32-bit", 4));
list.push_back(LoadStoreGroup::make<TIMER, store64_any>( "store/64-bit", 8));
list.push_back(LoadStoreGroup::make<TIMER, store128_any>("store/128-bit", 16));
list.push_back(LoadStoreGroup::make<TIMER, store256_any>("store/256-bit", 32));
list.push_back(LoadStoreGroup::make<TIMER, store512_any>("store/512-bit", 64, { x86Feature::AVX512F }));
#endif // #if !UARCH_BENCH_PORTABLE
}
#define REG_LOADSTORE(CLOCK) template void register_loadstore<CLOCK>(GroupList& list);
ALL_TIMERS_X(REG_LOADSTORE)