| /* | 
 |  * Copyright (C) 2021 The Android Open Source Project | 
 |  * | 
 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 |  * you may not use this file except in compliance with the License. | 
 |  * You may obtain a copy of the License at | 
 |  * | 
 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 |  * | 
 |  * Unless required by applicable law or agreed to in writing, software | 
 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 |  * See the License for the specific language governing permissions and | 
 |  * limitations under the License. | 
 |  */ | 
 |  | 
 | #include <binder/Parcel.h> | 
 | #include <benchmark/benchmark.h> | 
 |  | 
 | // Usage: atest binderParcelBenchmark | 
 |  | 
 | // For static assert(false) we need a template version to avoid early failure. | 
 | // See: https://stackoverflow.com/questions/51523965/template-dependent-false | 
 | template <typename T> | 
 | constexpr bool dependent_false_v = false; | 
 |  | 
 | template <template <typename ...> class V, typename T, typename... Args> | 
 | void writeVector(android::Parcel &p, const V<T, Args...> &v) { | 
 |     if constexpr (std::is_same_v<T, bool>) { | 
 |         p.writeBoolVector(v); | 
 |     } else if constexpr (std::is_same_v<T, uint8_t>) { | 
 |         p.writeByteVector(v); | 
 |     } else if constexpr (std::is_same_v<T, char16_t>) { | 
 |         p.writeCharVector(v); | 
 |     } else if constexpr (std::is_same_v<T, int32_t>) { | 
 |         p.writeInt32Vector(v); | 
 |     } else if constexpr (std::is_same_v<T, int64_t>) { | 
 |         p.writeInt64Vector(v); | 
 |     } else { | 
 |         static_assert(dependent_false_v<V<T>>); | 
 |     } | 
 | } | 
 |  | 
 | template <template <typename ...> class V, typename T, typename... Args> | 
 | void readVector(android::Parcel &p, V<T, Args...> *v) { | 
 |     if constexpr (std::is_same_v<T, bool>) { | 
 |         p.readBoolVector(v); | 
 |     } else if constexpr (std::is_same_v<T, uint8_t>) { | 
 |         p.readByteVector(v); | 
 |     } else if constexpr (std::is_same_v<T, char16_t>) { | 
 |         p.readCharVector(v); | 
 |     } else if constexpr (std::is_same_v<T, int32_t>) { | 
 |         p.readInt32Vector(v); | 
 |     } else if constexpr (std::is_same_v<T, int64_t>) { | 
 |         p.readInt64Vector(v); | 
 |     } else { | 
 |         static_assert(dependent_false_v<V<T>>); | 
 |     } | 
 | } | 
 |  | 
 | // Construct a series of args { 1 << 0, 1 << 1, ..., 1 << 10 } | 
 | static void VectorArgs(benchmark::internal::Benchmark* b) { | 
 |     for (int i = 0; i < 10; ++i) { | 
 |         b->Args({1 << i}); | 
 |     } | 
 | } | 
 |  | 
 | template <typename T> | 
 | static void BM_ParcelVector(benchmark::State& state) { | 
 |     const size_t elements = state.range(0); | 
 |  | 
 |     std::vector<T> v1(elements); | 
 |     std::vector<T> v2(elements); | 
 |     android::Parcel p; | 
 |     while (state.KeepRunning()) { | 
 |         p.setDataPosition(0); | 
 |         writeVector(p, v1); | 
 |  | 
 |         p.setDataPosition(0); | 
 |         readVector(p, &v2); | 
 |  | 
 |         benchmark::DoNotOptimize(v2[0]); | 
 |         benchmark::ClobberMemory(); | 
 |     } | 
 |     state.SetComplexityN(elements); | 
 | } | 
 |  | 
 | /* | 
 |   Parcel vector write than read. | 
 |   The read and write vectors are fixed, no resizing required. | 
 |  | 
 |   Results on Crosshatch Pixel 3XL | 
 |  | 
 |   #BM_BoolVector/1         44 ns      44 ns     15630626 | 
 |   #BM_BoolVector/2         54 ns      54 ns     12900340 | 
 |   #BM_BoolVector/4         73 ns      72 ns      9749841 | 
 |   #BM_BoolVector/8        107 ns     107 ns      6503326 | 
 |   #BM_BoolVector/16       186 ns     185 ns      3773627 | 
 |   #BM_BoolVector/32       337 ns     336 ns      2083877 | 
 |   #BM_BoolVector/64       607 ns     605 ns      1154113 | 
 |   #BM_BoolVector/128     1155 ns    1151 ns       608128 | 
 |   #BM_BoolVector/256     2259 ns    2253 ns       310973 | 
 |   #BM_BoolVector/512     4469 ns    4455 ns       157277 | 
 |   #BM_ByteVector/1         41 ns      41 ns     16837425 | 
 |   #BM_ByteVector/2         41 ns      41 ns     16820726 | 
 |   #BM_ByteVector/4         38 ns      38 ns     18217813 | 
 |   #BM_ByteVector/8         38 ns      38 ns     18290298 | 
 |   #BM_ByteVector/16        38 ns      38 ns     18117817 | 
 |   #BM_ByteVector/32        38 ns      38 ns     18172385 | 
 |   #BM_ByteVector/64        41 ns      41 ns     16950055 | 
 |   #BM_ByteVector/128       53 ns      53 ns     13170749 | 
 |   #BM_ByteVector/256       69 ns      69 ns     10113626 | 
 |   #BM_ByteVector/512      106 ns     106 ns      6561936 | 
 |   #BM_CharVector/1         38 ns      38 ns     18074831 | 
 |   #BM_CharVector/2         40 ns      40 ns     17206266 | 
 |   #BM_CharVector/4         50 ns      50 ns     13785944 | 
 |   #BM_CharVector/8         67 ns      67 ns     10223316 | 
 |   #BM_CharVector/16        96 ns      96 ns      7297285 | 
 |   #BM_CharVector/32       156 ns     155 ns      4484845 | 
 |   #BM_CharVector/64       277 ns     276 ns      2536003 | 
 |   #BM_CharVector/128      520 ns     518 ns      1347070 | 
 |   #BM_CharVector/256     1006 ns    1003 ns       695952 | 
 |   #BM_CharVector/512     1976 ns    1970 ns       354673 | 
 |   #BM_Int32Vector/1        41 ns      41 ns     16951262 | 
 |   #BM_Int32Vector/2        41 ns      41 ns     16916883 | 
 |   #BM_Int32Vector/4        41 ns      41 ns     16761373 | 
 |   #BM_Int32Vector/8        42 ns      42 ns     16553179 | 
 |   #BM_Int32Vector/16       43 ns      43 ns     16200362 | 
 |   #BM_Int32Vector/32       55 ns      54 ns     12724454 | 
 |   #BM_Int32Vector/64       70 ns      69 ns     10049223 | 
 |   #BM_Int32Vector/128     107 ns     107 ns      6525796 | 
 |   #BM_Int32Vector/256     179 ns     178 ns      3922563 | 
 |   #BM_Int32Vector/512     324 ns     323 ns      2160653 | 
 |   #BM_Int64Vector/1        41 ns      41 ns     16909470 | 
 |   #BM_Int64Vector/2        41 ns      41 ns     16740788 | 
 |   #BM_Int64Vector/4        42 ns      42 ns     16564197 | 
 |   #BM_Int64Vector/8        43 ns      42 ns     16284082 | 
 |   #BM_Int64Vector/16       54 ns      54 ns     12839474 | 
 |   #BM_Int64Vector/32       69 ns      69 ns     10011010 | 
 |   #BM_Int64Vector/64      107 ns     106 ns      6557956 | 
 |   #BM_Int64Vector/128     177 ns     177 ns      3925618 | 
 |   #BM_Int64Vector/256     324 ns     323 ns      2163321 | 
 |   #BM_Int64Vector/512     613 ns     611 ns      1140418 | 
 | */ | 
 |  | 
 | static void BM_BoolVector(benchmark::State& state) { | 
 |     BM_ParcelVector<bool>(state); | 
 | } | 
 |  | 
 | static void BM_ByteVector(benchmark::State& state) { | 
 |     BM_ParcelVector<uint8_t>(state); | 
 | } | 
 |  | 
 | static void BM_CharVector(benchmark::State& state) { | 
 |     BM_ParcelVector<char16_t>(state); | 
 | } | 
 |  | 
 | static void BM_Int32Vector(benchmark::State& state) { | 
 |     BM_ParcelVector<int32_t>(state); | 
 | } | 
 |  | 
 | static void BM_Int64Vector(benchmark::State& state) { | 
 |     BM_ParcelVector<int64_t>(state); | 
 | } | 
 |  | 
 | BENCHMARK(BM_BoolVector)->Apply(VectorArgs); | 
 | BENCHMARK(BM_ByteVector)->Apply(VectorArgs); | 
 | BENCHMARK(BM_CharVector)->Apply(VectorArgs); | 
 | BENCHMARK(BM_Int32Vector)->Apply(VectorArgs); | 
 | BENCHMARK(BM_Int64Vector)->Apply(VectorArgs); | 
 |  | 
 | BENCHMARK_MAIN(); |