| /* |
| * Copyright (C) 2016 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. |
| */ |
| |
| #define LOG_TAG "libhwbinder_benchmark" |
| |
| #include <sys/types.h> |
| #include <sys/wait.h> |
| #include <unistd.h> |
| |
| #include <iostream> |
| |
| #include <log/log.h> |
| #include <utils/StrongPointer.h> |
| |
| #include <benchmark/benchmark.h> |
| #include <hidl/Status.h> |
| #include <hidl/ServiceManagement.h> |
| |
| #include <android/hardware/tests/libhwbinder/1.0/IBenchmark.h> |
| |
| // libutils: |
| using android::OK; |
| using android::sp; |
| using android::status_t; |
| |
| // libhidl: |
| using android::hardware::defaultServiceManager; |
| using android::hardware::Return; |
| using android::hardware::Void; |
| using android::hardware::hidl_vec; |
| |
| // Standard library |
| using std::cerr; |
| using std::cout; |
| using std::endl; |
| using std::string; |
| using std::unique_ptr; |
| using std::vector; |
| |
| // Generated HIDL files |
| using android::hardware::tests::libhwbinder::V1_0::IBenchmark; |
| |
| const char gServiceName[] = "android.hardware.tests.libhwbinder.IBenchmark"; |
| |
| static bool startServer() { |
| sp<IBenchmark> service = IBenchmark::getService(gServiceName, true); |
| status_t status = service->registerAsService(gServiceName); |
| |
| if (status != ::android::OK) { |
| ALOGE("Failed to register service %s.", gServiceName); |
| exit(EXIT_FAILURE); |
| } |
| |
| return 0; |
| } |
| |
| static void BM_sendVec(benchmark::State& state, sp<IBenchmark> service) { |
| // Prepare data to IPC |
| hidl_vec<uint8_t> data_vec; |
| data_vec.resize(state.range(0)); |
| for (int i = 0; i < state.range(0); i++) { |
| data_vec[i] = i % 256; |
| } |
| // Start running |
| while (state.KeepRunning()) { |
| service->sendVec(data_vec, [&] (const auto &/*res*/) { |
| }); |
| } |
| } |
| |
| static void BM_sendVec_passthrough(benchmark::State& state) { |
| // getService automatically retries |
| sp<IBenchmark> service = IBenchmark::getService(gServiceName, true /* getStub */); |
| if (service == nullptr) { |
| state.SkipWithError("Failed to retrieve benchmark service."); |
| } |
| if (service->isRemote()) { |
| state.SkipWithError("Benchmark service is remote."); |
| } |
| BM_sendVec(state, service); |
| } |
| |
| static void BM_sendVec_binderize(benchmark::State& state) { |
| // getService automatically retries |
| sp<IBenchmark> service = IBenchmark::getService(gServiceName); |
| if (service == nullptr) { |
| state.SkipWithError("Failed to retrieve benchmark service."); |
| } |
| if (!service->isRemote()) { |
| state.SkipWithError("Unable to fetch remote benchmark service."); |
| } |
| BM_sendVec(state, service); |
| } |
| |
| int main(int argc, char* argv []) { |
| setenv("TREBLE_TESTING_OVERRIDE", "true", true); |
| |
| enum HwBinderMode { |
| kBinderize = 0, |
| kPassthrough = 1, |
| }; |
| HwBinderMode mode = HwBinderMode::kBinderize; |
| |
| // Parse arguments. |
| for (int i = 1; i < argc; i++) { |
| if (string(argv[i]) == "-m") { |
| if (!strcmp(argv[i + 1], "PASSTHROUGH")) { |
| mode = HwBinderMode::kPassthrough; |
| } |
| break; |
| } |
| } |
| if (mode == HwBinderMode::kBinderize) { |
| BENCHMARK(BM_sendVec_binderize)->RangeMultiplier(2)->Range(4, 65536); |
| } else { |
| BENCHMARK(BM_sendVec_passthrough)->RangeMultiplier(2)->Range(4, 65536); |
| } |
| |
| ::benchmark::Initialize(&argc, argv); |
| |
| pid_t pid = fork(); |
| if (pid == 0) { |
| // Child, start benchmarks |
| ::benchmark::RunSpecifiedBenchmarks(); |
| } else { |
| startServer(); |
| while (true) { |
| int stat, retval; |
| retval = wait(&stat); |
| if (retval == -1 && errno == ECHILD) { |
| break; |
| } |
| } |
| }; |
| } |