1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2013 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "bench/Benchmark.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTemplates.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkTSort.h"
13*c8dee2aaSAndroid Build Coastguard Worker
14*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
15*c8dee2aaSAndroid Build Coastguard Worker #include <stdlib.h>
16*c8dee2aaSAndroid Build Coastguard Worker
17*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
18*c8dee2aaSAndroid Build Coastguard Worker
19*c8dee2aaSAndroid Build Coastguard Worker static const int N = 1000;
20*c8dee2aaSAndroid Build Coastguard Worker
rand_proc(int array[N])21*c8dee2aaSAndroid Build Coastguard Worker static void rand_proc(int array[N]) {
22*c8dee2aaSAndroid Build Coastguard Worker SkRandom rand;
23*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < N; ++i) {
24*c8dee2aaSAndroid Build Coastguard Worker array[i] = rand.nextS();
25*c8dee2aaSAndroid Build Coastguard Worker }
26*c8dee2aaSAndroid Build Coastguard Worker }
27*c8dee2aaSAndroid Build Coastguard Worker
randN_proc(int array[N])28*c8dee2aaSAndroid Build Coastguard Worker static void randN_proc(int array[N]) {
29*c8dee2aaSAndroid Build Coastguard Worker SkRandom rand;
30*c8dee2aaSAndroid Build Coastguard Worker int mod = N / 10;
31*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < N; ++i) {
32*c8dee2aaSAndroid Build Coastguard Worker array[i] = rand.nextU() % mod;
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker }
35*c8dee2aaSAndroid Build Coastguard Worker
forward_proc(int array[N])36*c8dee2aaSAndroid Build Coastguard Worker static void forward_proc(int array[N]) {
37*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < N; ++i) {
38*c8dee2aaSAndroid Build Coastguard Worker array[i] = i;
39*c8dee2aaSAndroid Build Coastguard Worker }
40*c8dee2aaSAndroid Build Coastguard Worker }
41*c8dee2aaSAndroid Build Coastguard Worker
backward_proc(int array[N])42*c8dee2aaSAndroid Build Coastguard Worker static void backward_proc(int array[N]) {
43*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < N; ++i) {
44*c8dee2aaSAndroid Build Coastguard Worker array[i] = -i;
45*c8dee2aaSAndroid Build Coastguard Worker }
46*c8dee2aaSAndroid Build Coastguard Worker }
47*c8dee2aaSAndroid Build Coastguard Worker
same_proc(int array[N])48*c8dee2aaSAndroid Build Coastguard Worker static void same_proc(int array[N]) {
49*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < N; ++i) {
50*c8dee2aaSAndroid Build Coastguard Worker array[i] = N;
51*c8dee2aaSAndroid Build Coastguard Worker }
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker
54*c8dee2aaSAndroid Build Coastguard Worker typedef void (*SortProc)(int array[N]);
55*c8dee2aaSAndroid Build Coastguard Worker
56*c8dee2aaSAndroid Build Coastguard Worker enum Type {
57*c8dee2aaSAndroid Build Coastguard Worker kRand, kRandN, kFore, kBack, kSame
58*c8dee2aaSAndroid Build Coastguard Worker };
59*c8dee2aaSAndroid Build Coastguard Worker
60*c8dee2aaSAndroid Build Coastguard Worker static const struct {
61*c8dee2aaSAndroid Build Coastguard Worker const char* fName;
62*c8dee2aaSAndroid Build Coastguard Worker SortProc fProc;
63*c8dee2aaSAndroid Build Coastguard Worker } gRec[] = {
64*c8dee2aaSAndroid Build Coastguard Worker { "rand", rand_proc },
65*c8dee2aaSAndroid Build Coastguard Worker { "rand10", randN_proc },
66*c8dee2aaSAndroid Build Coastguard Worker { "forward", forward_proc },
67*c8dee2aaSAndroid Build Coastguard Worker { "backward", backward_proc },
68*c8dee2aaSAndroid Build Coastguard Worker { "repeated", same_proc },
69*c8dee2aaSAndroid Build Coastguard Worker };
70*c8dee2aaSAndroid Build Coastguard Worker
skqsort_sort(int array[N])71*c8dee2aaSAndroid Build Coastguard Worker static void skqsort_sort(int array[N]) {
72*c8dee2aaSAndroid Build Coastguard Worker SkTQSort<int>(array, array + N);
73*c8dee2aaSAndroid Build Coastguard Worker }
74*c8dee2aaSAndroid Build Coastguard Worker
skheap_sort(int array[N])75*c8dee2aaSAndroid Build Coastguard Worker static void skheap_sort(int array[N]) {
76*c8dee2aaSAndroid Build Coastguard Worker SkTHeapSort<int>(array, N);
77*c8dee2aaSAndroid Build Coastguard Worker }
78*c8dee2aaSAndroid Build Coastguard Worker
79*c8dee2aaSAndroid Build Coastguard Worker extern "C" {
int_compare(const void * a,const void * b)80*c8dee2aaSAndroid Build Coastguard Worker static int int_compare(const void* a, const void* b) {
81*c8dee2aaSAndroid Build Coastguard Worker const int ai = *(const int*)a;
82*c8dee2aaSAndroid Build Coastguard Worker const int bi = *(const int*)b;
83*c8dee2aaSAndroid Build Coastguard Worker return ai < bi ? -1 : (ai > bi);
84*c8dee2aaSAndroid Build Coastguard Worker }
85*c8dee2aaSAndroid Build Coastguard Worker }
86*c8dee2aaSAndroid Build Coastguard Worker
qsort_sort(int array[N])87*c8dee2aaSAndroid Build Coastguard Worker static void qsort_sort(int array[N]) {
88*c8dee2aaSAndroid Build Coastguard Worker qsort(array, N, sizeof(int), int_compare);
89*c8dee2aaSAndroid Build Coastguard Worker }
90*c8dee2aaSAndroid Build Coastguard Worker
stdsort_sort(int array[N])91*c8dee2aaSAndroid Build Coastguard Worker static void stdsort_sort(int array[N]) {
92*c8dee2aaSAndroid Build Coastguard Worker std::sort(array, array+N);
93*c8dee2aaSAndroid Build Coastguard Worker }
94*c8dee2aaSAndroid Build Coastguard Worker
95*c8dee2aaSAndroid Build Coastguard Worker enum SortType {
96*c8dee2aaSAndroid Build Coastguard Worker kSKQSort, kSKHeap, kQSort, kStdSort,
97*c8dee2aaSAndroid Build Coastguard Worker };
98*c8dee2aaSAndroid Build Coastguard Worker
99*c8dee2aaSAndroid Build Coastguard Worker static const struct {
100*c8dee2aaSAndroid Build Coastguard Worker const char* fName;
101*c8dee2aaSAndroid Build Coastguard Worker SortProc fProc;
102*c8dee2aaSAndroid Build Coastguard Worker } gSorts[] = {
103*c8dee2aaSAndroid Build Coastguard Worker { "skqsort", skqsort_sort },
104*c8dee2aaSAndroid Build Coastguard Worker { "skheap", skheap_sort },
105*c8dee2aaSAndroid Build Coastguard Worker { "qsort", qsort_sort },
106*c8dee2aaSAndroid Build Coastguard Worker { "stdsort", stdsort_sort },
107*c8dee2aaSAndroid Build Coastguard Worker };
108*c8dee2aaSAndroid Build Coastguard Worker
109*c8dee2aaSAndroid Build Coastguard Worker class SortBench : public Benchmark {
110*c8dee2aaSAndroid Build Coastguard Worker SkString fName;
111*c8dee2aaSAndroid Build Coastguard Worker const Type fType;
112*c8dee2aaSAndroid Build Coastguard Worker const SortProc fSortProc;
113*c8dee2aaSAndroid Build Coastguard Worker AutoTMalloc<int> fUnsorted;
114*c8dee2aaSAndroid Build Coastguard Worker
115*c8dee2aaSAndroid Build Coastguard Worker public:
SortBench(Type t,SortType s)116*c8dee2aaSAndroid Build Coastguard Worker SortBench(Type t, SortType s) : fType(t), fSortProc(gSorts[s].fProc) {
117*c8dee2aaSAndroid Build Coastguard Worker fName.printf("sort_%s_%s", gSorts[s].fName, gRec[t].fName);
118*c8dee2aaSAndroid Build Coastguard Worker }
119*c8dee2aaSAndroid Build Coastguard Worker
isSuitableFor(Backend backend)120*c8dee2aaSAndroid Build Coastguard Worker bool isSuitableFor(Backend backend) override {
121*c8dee2aaSAndroid Build Coastguard Worker return backend == Backend::kNonRendering;
122*c8dee2aaSAndroid Build Coastguard Worker }
123*c8dee2aaSAndroid Build Coastguard Worker
124*c8dee2aaSAndroid Build Coastguard Worker protected:
onGetName()125*c8dee2aaSAndroid Build Coastguard Worker const char* onGetName() override {
126*c8dee2aaSAndroid Build Coastguard Worker return fName.c_str();
127*c8dee2aaSAndroid Build Coastguard Worker }
128*c8dee2aaSAndroid Build Coastguard Worker
129*c8dee2aaSAndroid Build Coastguard Worker // Delayed initialization only done if onDraw will be called.
onDelayedSetup()130*c8dee2aaSAndroid Build Coastguard Worker void onDelayedSetup() override {
131*c8dee2aaSAndroid Build Coastguard Worker fUnsorted.reset(N);
132*c8dee2aaSAndroid Build Coastguard Worker gRec[fType].fProc(fUnsorted.get());
133*c8dee2aaSAndroid Build Coastguard Worker }
134*c8dee2aaSAndroid Build Coastguard Worker
onDraw(int loops,SkCanvas *)135*c8dee2aaSAndroid Build Coastguard Worker void onDraw(int loops, SkCanvas*) override {
136*c8dee2aaSAndroid Build Coastguard Worker AutoTMalloc<int> sorted(N);
137*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < loops; i++) {
138*c8dee2aaSAndroid Build Coastguard Worker memcpy(sorted.get(), fUnsorted.get(), N*sizeof(int));
139*c8dee2aaSAndroid Build Coastguard Worker fSortProc(sorted.get());
140*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_DEBUG
141*c8dee2aaSAndroid Build Coastguard Worker for (int j = 1; j < N; ++j) {
142*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(sorted[j - 1] <= sorted[j]);
143*c8dee2aaSAndroid Build Coastguard Worker }
144*c8dee2aaSAndroid Build Coastguard Worker #endif
145*c8dee2aaSAndroid Build Coastguard Worker }
146*c8dee2aaSAndroid Build Coastguard Worker }
147*c8dee2aaSAndroid Build Coastguard Worker
148*c8dee2aaSAndroid Build Coastguard Worker private:
149*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = Benchmark;
150*c8dee2aaSAndroid Build Coastguard Worker };
151*c8dee2aaSAndroid Build Coastguard Worker
152*c8dee2aaSAndroid Build Coastguard Worker ///////////////////////////////////////////////////////////////////////////////
153*c8dee2aaSAndroid Build Coastguard Worker
NewSkQSort(Type t)154*c8dee2aaSAndroid Build Coastguard Worker static Benchmark* NewSkQSort(Type t) {
155*c8dee2aaSAndroid Build Coastguard Worker return new SortBench(t, kSKQSort);
156*c8dee2aaSAndroid Build Coastguard Worker }
NewSkHeap(Type t)157*c8dee2aaSAndroid Build Coastguard Worker static Benchmark* NewSkHeap(Type t) {
158*c8dee2aaSAndroid Build Coastguard Worker return new SortBench(t, kSKHeap);
159*c8dee2aaSAndroid Build Coastguard Worker }
NewQSort(Type t)160*c8dee2aaSAndroid Build Coastguard Worker static Benchmark* NewQSort(Type t) {
161*c8dee2aaSAndroid Build Coastguard Worker return new SortBench(t, kQSort);
162*c8dee2aaSAndroid Build Coastguard Worker }
NewStdSort(Type t)163*c8dee2aaSAndroid Build Coastguard Worker static Benchmark* NewStdSort(Type t) {
164*c8dee2aaSAndroid Build Coastguard Worker return new SortBench(t, kStdSort);
165*c8dee2aaSAndroid Build Coastguard Worker }
166*c8dee2aaSAndroid Build Coastguard Worker
167*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkQSort(kRand); )
168*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkHeap(kRand); )
169*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewQSort(kRand); )
170*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewStdSort(kRand); )
171*c8dee2aaSAndroid Build Coastguard Worker
172*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkQSort(kRandN); )
173*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkHeap(kRandN); )
174*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewQSort(kRandN); )
175*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewStdSort(kRandN); )
176*c8dee2aaSAndroid Build Coastguard Worker
177*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkQSort(kFore); )
178*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkHeap(kFore); )
179*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewQSort(kFore); )
180*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewStdSort(kFore); )
181*c8dee2aaSAndroid Build Coastguard Worker
182*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkQSort(kBack); )
183*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkHeap(kBack); )
184*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewQSort(kBack); )
185*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewStdSort(kBack); )
186*c8dee2aaSAndroid Build Coastguard Worker
187*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkQSort(kSame); )
188*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewSkHeap(kSame); )
189*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewQSort(kSame); )
190*c8dee2aaSAndroid Build Coastguard Worker DEF_BENCH( return NewStdSort(kSame); )
191