1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <pthread.h>
18 
19 #include <benchmark/benchmark.h>
20 #include "util.h"
21 
22 // Stop GCC optimizing out our pure function.
23 /* Must not be static! */ pthread_t (*pthread_self_fp)() = pthread_self;
24 
BM_pthread_self(benchmark::State & state)25 static void BM_pthread_self(benchmark::State& state) {
26   while (state.KeepRunning()) {
27     pthread_self_fp();
28   }
29 }
30 BIONIC_BENCHMARK(BM_pthread_self);
31 
BM_pthread_getspecific(benchmark::State & state)32 static void BM_pthread_getspecific(benchmark::State& state) {
33   pthread_key_t key;
34   pthread_key_create(&key, nullptr);
35 
36   while (state.KeepRunning()) {
37     pthread_getspecific(key);
38   }
39 
40   pthread_key_delete(key);
41 }
42 BIONIC_BENCHMARK(BM_pthread_getspecific);
43 
BM_pthread_setspecific(benchmark::State & state)44 static void BM_pthread_setspecific(benchmark::State& state) {
45   pthread_key_t key;
46   pthread_key_create(&key, nullptr);
47 
48   while (state.KeepRunning()) {
49     pthread_setspecific(key, nullptr);
50   }
51 
52   pthread_key_delete(key);
53 }
54 BIONIC_BENCHMARK(BM_pthread_setspecific);
55 
NoOpPthreadOnceInitFunction()56 static void NoOpPthreadOnceInitFunction() {}
57 
BM_pthread_once(benchmark::State & state)58 static void BM_pthread_once(benchmark::State& state) {
59   static pthread_once_t once = PTHREAD_ONCE_INIT;
60   pthread_once(&once, NoOpPthreadOnceInitFunction);
61 
62   while (state.KeepRunning()) {
63     pthread_once(&once, NoOpPthreadOnceInitFunction);
64   }
65 }
66 BIONIC_BENCHMARK(BM_pthread_once);
67 
BM_pthread_mutex_lock(benchmark::State & state)68 static void BM_pthread_mutex_lock(benchmark::State& state) {
69   pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
70 
71   while (state.KeepRunning()) {
72     pthread_mutex_lock(&mutex);
73     pthread_mutex_unlock(&mutex);
74   }
75 }
76 BIONIC_BENCHMARK(BM_pthread_mutex_lock);
77 
78 #if !defined(ANDROID_HOST_MUSL)
BM_pthread_mutex_lock_ERRORCHECK(benchmark::State & state)79 static void BM_pthread_mutex_lock_ERRORCHECK(benchmark::State& state) {
80   pthread_mutex_t mutex = PTHREAD_ERRORCHECK_MUTEX_INITIALIZER_NP;
81 
82   while (state.KeepRunning()) {
83     pthread_mutex_lock(&mutex);
84     pthread_mutex_unlock(&mutex);
85   }
86 }
87 BIONIC_BENCHMARK(BM_pthread_mutex_lock_ERRORCHECK);
88 #endif
89 
90 #if !defined(ANDROID_HOST_MUSL)
BM_pthread_mutex_lock_RECURSIVE(benchmark::State & state)91 static void BM_pthread_mutex_lock_RECURSIVE(benchmark::State& state) {
92   pthread_mutex_t mutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
93 
94   while (state.KeepRunning()) {
95     pthread_mutex_lock(&mutex);
96     pthread_mutex_unlock(&mutex);
97   }
98 }
99 BIONIC_BENCHMARK(BM_pthread_mutex_lock_RECURSIVE);
100 #endif
101 
102 namespace {
103 struct PIMutex {
104   pthread_mutex_t mutex;
105 
PIMutex__anon218be1320111::PIMutex106   explicit PIMutex(int type) {
107     pthread_mutexattr_t attr;
108     pthread_mutexattr_init(&attr);
109     pthread_mutexattr_settype(&attr, type);
110     pthread_mutexattr_setprotocol(&attr, PTHREAD_PRIO_INHERIT);
111     pthread_mutex_init(&mutex, &attr);
112     pthread_mutexattr_destroy(&attr);
113   }
114 
~PIMutex__anon218be1320111::PIMutex115   ~PIMutex() {
116     pthread_mutex_destroy(&mutex);
117   }
118 };
119 }
120 
BM_pthread_mutex_lock_PI(benchmark::State & state)121 static void BM_pthread_mutex_lock_PI(benchmark::State& state) {
122   PIMutex m(PTHREAD_MUTEX_NORMAL);
123 
124   while (state.KeepRunning()) {
125     pthread_mutex_lock(&m.mutex);
126     pthread_mutex_unlock(&m.mutex);
127   }
128 }
129 BIONIC_BENCHMARK(BM_pthread_mutex_lock_PI);
130 
BM_pthread_mutex_lock_ERRORCHECK_PI(benchmark::State & state)131 static void BM_pthread_mutex_lock_ERRORCHECK_PI(benchmark::State& state) {
132   PIMutex m(PTHREAD_MUTEX_ERRORCHECK);
133 
134   while (state.KeepRunning()) {
135     pthread_mutex_lock(&m.mutex);
136     pthread_mutex_unlock(&m.mutex);
137   }
138 }
139 BIONIC_BENCHMARK(BM_pthread_mutex_lock_ERRORCHECK_PI);
140 
BM_pthread_mutex_lock_RECURSIVE_PI(benchmark::State & state)141 static void BM_pthread_mutex_lock_RECURSIVE_PI(benchmark::State& state) {
142   PIMutex m(PTHREAD_MUTEX_RECURSIVE);
143 
144   while (state.KeepRunning()) {
145     pthread_mutex_lock(&m.mutex);
146     pthread_mutex_unlock(&m.mutex);
147   }
148 }
149 BIONIC_BENCHMARK(BM_pthread_mutex_lock_RECURSIVE_PI);
150 
BM_pthread_rwlock_read(benchmark::State & state)151 static void BM_pthread_rwlock_read(benchmark::State& state) {
152   pthread_rwlock_t lock;
153   pthread_rwlock_init(&lock, nullptr);
154 
155   while (state.KeepRunning()) {
156     pthread_rwlock_rdlock(&lock);
157     pthread_rwlock_unlock(&lock);
158   }
159 
160   pthread_rwlock_destroy(&lock);
161 }
162 BIONIC_BENCHMARK(BM_pthread_rwlock_read);
163 
BM_pthread_rwlock_write(benchmark::State & state)164 static void BM_pthread_rwlock_write(benchmark::State& state) {
165   pthread_rwlock_t lock;
166   pthread_rwlock_init(&lock, nullptr);
167 
168   while (state.KeepRunning()) {
169     pthread_rwlock_wrlock(&lock);
170     pthread_rwlock_unlock(&lock);
171   }
172 
173   pthread_rwlock_destroy(&lock);
174 }
175 BIONIC_BENCHMARK(BM_pthread_rwlock_write);
176 
IdleThread(void *)177 static void* IdleThread(void*) {
178   return nullptr;
179 }
180 
BM_pthread_create(benchmark::State & state)181 static void BM_pthread_create(benchmark::State& state) {
182   while (state.KeepRunning()) {
183     pthread_t thread;
184     pthread_create(&thread, nullptr, IdleThread, nullptr);
185     state.PauseTiming();
186     pthread_join(thread, nullptr);
187     state.ResumeTiming();
188   }
189 }
190 BIONIC_BENCHMARK(BM_pthread_create);
191 
RunThread(void *)192 static void* RunThread(void*) {
193   return nullptr;
194 }
195 
BM_pthread_create_and_run(benchmark::State & state)196 static void BM_pthread_create_and_run(benchmark::State& state) {
197   while (state.KeepRunning()) {
198     pthread_t thread;
199     pthread_create(&thread, nullptr, RunThread, &state);
200     pthread_join(thread, nullptr);
201   }
202 }
203 BIONIC_BENCHMARK(BM_pthread_create_and_run);
204 
ExitThread(void *)205 static void* ExitThread(void*) {
206   pthread_exit(nullptr);
207 }
208 
BM_pthread_exit_and_join(benchmark::State & state)209 static void BM_pthread_exit_and_join(benchmark::State& state) {
210   while (state.KeepRunning()) {
211     pthread_t thread;
212     pthread_create(&thread, nullptr, ExitThread, nullptr);
213     pthread_join(thread, nullptr);
214   }
215 }
216 BIONIC_BENCHMARK(BM_pthread_exit_and_join);
217 
BM_pthread_key_create(benchmark::State & state)218 static void BM_pthread_key_create(benchmark::State& state) {
219   while (state.KeepRunning()) {
220     pthread_key_t key;
221     pthread_key_create(&key, nullptr);
222 
223     state.PauseTiming();
224     pthread_key_delete(key);
225     state.ResumeTiming();
226   }
227 }
228 BIONIC_BENCHMARK(BM_pthread_key_create);
229 
BM_pthread_key_delete(benchmark::State & state)230 static void BM_pthread_key_delete(benchmark::State& state) {
231   while (state.KeepRunning()) {
232     state.PauseTiming();
233     pthread_key_t key;
234     pthread_key_create(&key, nullptr);
235     state.ResumeTiming();
236 
237     pthread_key_delete(key);
238   }
239 }
240 BIONIC_BENCHMARK(BM_pthread_key_delete);
241