|  | /* | 
|  | * 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. | 
|  | */ | 
|  |  | 
|  | #include <gtest/gtest.h> | 
|  |  | 
|  | #include <condition_variable> | 
|  | #include <mutex> | 
|  | #include <thread> | 
|  |  | 
|  | #include "socket.h" | 
|  | #include "sysdeps.h" | 
|  | #include "sysdeps/chrono.h" | 
|  |  | 
|  | static void WaitForFdeventLoop() { | 
|  | // Sleep for a bit to make sure that network events have propagated. | 
|  | std::this_thread::sleep_for(100ms); | 
|  |  | 
|  | // fdevent_run_on_main_thread has a guaranteed ordering, and is guaranteed to happen after | 
|  | // socket events, so as soon as our function is called, we know that we've processed all | 
|  | // previous events. | 
|  | std::mutex mutex; | 
|  | std::condition_variable cv; | 
|  | std::unique_lock<std::mutex> lock(mutex); | 
|  | fdevent_run_on_main_thread([&]() { | 
|  | mutex.lock(); | 
|  | mutex.unlock(); | 
|  | cv.notify_one(); | 
|  | }); | 
|  | cv.wait(lock); | 
|  | } | 
|  |  | 
|  | class FdeventTest : public ::testing::Test { | 
|  | protected: | 
|  | int dummy = -1; | 
|  |  | 
|  | static void SetUpTestCase() { | 
|  | #if !defined(_WIN32) | 
|  | ASSERT_NE(SIG_ERR, signal(SIGPIPE, SIG_IGN)); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void SetUp() override { | 
|  | fdevent_reset(); | 
|  | ASSERT_EQ(0u, fdevent_installed_count()); | 
|  | } | 
|  |  | 
|  | // Register a dummy socket used to wake up the fdevent loop to tell it to die. | 
|  | void PrepareThread() { | 
|  | int dummy_fds[2]; | 
|  | if (adb_socketpair(dummy_fds) != 0) { | 
|  | FAIL() << "failed to create socketpair: " << strerror(errno); | 
|  | } | 
|  |  | 
|  | asocket* dummy_socket = create_local_socket(dummy_fds[1]); | 
|  | if (!dummy_socket) { | 
|  | FAIL() << "failed to create local socket: " << strerror(errno); | 
|  | } | 
|  | dummy_socket->ready(dummy_socket); | 
|  | dummy = dummy_fds[0]; | 
|  |  | 
|  | thread_ = std::thread([]() { fdevent_loop(); }); | 
|  | WaitForFdeventLoop(); | 
|  | } | 
|  |  | 
|  | size_t GetAdditionalLocalSocketCount() { | 
|  | // dummy socket installed in PrepareThread() + fdevent_run_on_main_thread socket | 
|  | return 2; | 
|  | } | 
|  |  | 
|  | void TerminateThread() { | 
|  | fdevent_terminate_loop(); | 
|  | ASSERT_TRUE(WriteFdExactly(dummy, "", 1)); | 
|  | thread_.join(); | 
|  | ASSERT_EQ(0, adb_close(dummy)); | 
|  | } | 
|  |  | 
|  | std::thread thread_; | 
|  | }; |