//===-- MainLoopTest.cpp ----------------------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "lldb/Host/MainLoop.h" #include "lldb/Host/ConnectionFileDescriptor.h" #include "lldb/Host/PseudoTerminal.h" #include "lldb/Host/common/TCPSocket.h" #include "gtest/gtest.h" #include using namespace lldb_private; namespace { class MainLoopTest : public testing::Test { public: static void SetUpTestCase() { #ifdef _MSC_VER WSADATA data; ASSERT_EQ(0, WSAStartup(MAKEWORD(2, 2), &data)); #endif } static void TearDownTestCase() { #ifdef _MSC_VER ASSERT_EQ(0, WSACleanup()); #endif } void SetUp() override { bool child_processes_inherit = false; Status error; std::unique_ptr listen_socket_up( new TCPSocket(true, child_processes_inherit)); ASSERT_TRUE(error.Success()); error = listen_socket_up->Listen("localhost:0", 5); ASSERT_TRUE(error.Success()); Socket *accept_socket; std::future accept_error = std::async(std::launch::async, [&] { return listen_socket_up->Accept(accept_socket); }); std::unique_ptr connect_socket_up( new TCPSocket(true, child_processes_inherit)); error = connect_socket_up->Connect( llvm::formatv("localhost:{0}", listen_socket_up->GetLocalPortNumber()) .str()); ASSERT_TRUE(error.Success()); ASSERT_TRUE(accept_error.get().Success()); callback_count = 0; socketpair[0] = std::move(connect_socket_up); socketpair[1].reset(accept_socket); } void TearDown() override { socketpair[0].reset(); socketpair[1].reset(); } protected: MainLoop::Callback make_callback() { return [&](MainLoopBase &loop) { ++callback_count; loop.RequestTermination(); }; } std::shared_ptr socketpair[2]; unsigned callback_count; }; } // namespace TEST_F(MainLoopTest, ReadObject) { char X = 'X'; size_t len = sizeof(X); ASSERT_TRUE(socketpair[0]->Write(&X, len).Success()); MainLoop loop; Status error; auto handle = loop.RegisterReadObject(socketpair[1], make_callback(), error); ASSERT_TRUE(error.Success()); ASSERT_TRUE(handle); ASSERT_TRUE(loop.Run().Success()); ASSERT_EQ(1u, callback_count); } TEST_F(MainLoopTest, TerminatesImmediately) { char X = 'X'; size_t len = sizeof(X); ASSERT_TRUE(socketpair[0]->Write(&X, len).Success()); ASSERT_TRUE(socketpair[1]->Write(&X, len).Success()); MainLoop loop; Status error; auto handle0 = loop.RegisterReadObject(socketpair[0], make_callback(), error); ASSERT_TRUE(error.Success()); auto handle1 = loop.RegisterReadObject(socketpair[1], make_callback(), error); ASSERT_TRUE(error.Success()); ASSERT_TRUE(loop.Run().Success()); ASSERT_EQ(1u, callback_count); } #ifdef LLVM_ON_UNIX TEST_F(MainLoopTest, DetectsEOF) { PseudoTerminal term; ASSERT_TRUE(term.OpenFirstAvailableMaster(O_RDWR, nullptr, 0)); ASSERT_TRUE(term.OpenSlave(O_RDWR | O_NOCTTY, nullptr, 0)); auto conn = llvm::make_unique( term.ReleaseMasterFileDescriptor(), true); Status error; MainLoop loop; auto handle = loop.RegisterReadObject(conn->GetReadObject(), make_callback(), error); ASSERT_TRUE(error.Success()); term.CloseSlaveFileDescriptor(); ASSERT_TRUE(loop.Run().Success()); ASSERT_EQ(1u, callback_count); } TEST_F(MainLoopTest, Signal) { MainLoop loop; Status error; auto handle = loop.RegisterSignal(SIGUSR1, make_callback(), error); ASSERT_TRUE(error.Success()); kill(getpid(), SIGUSR1); ASSERT_TRUE(loop.Run().Success()); ASSERT_EQ(1u, callback_count); } #endif