2017-05-04 18:11:33 +08:00
|
|
|
//===-- 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"
|
2017-12-11 17:33:18 +08:00
|
|
|
#include "lldb/Host/ConnectionFileDescriptor.h"
|
|
|
|
#include "lldb/Host/PseudoTerminal.h"
|
2017-05-04 18:11:33 +08:00
|
|
|
#include "lldb/Host/common/TCPSocket.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
#include <future>
|
|
|
|
|
|
|
|
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;
|
2017-05-12 12:51:55 +08:00
|
|
|
Status error;
|
2017-05-04 18:11:33 +08:00
|
|
|
std::unique_ptr<TCPSocket> 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;
|
2017-05-12 12:51:55 +08:00
|
|
|
std::future<Status> accept_error = std::async(std::launch::async, [&] {
|
2017-05-04 18:11:33 +08:00
|
|
|
return listen_socket_up->Accept(accept_socket);
|
|
|
|
});
|
|
|
|
|
|
|
|
std::unique_ptr<TCPSocket> 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<Socket> 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;
|
|
|
|
|
2017-05-12 12:51:55 +08:00
|
|
|
Status error;
|
2017-05-04 18:11:33 +08:00
|
|
|
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;
|
2017-05-12 12:51:55 +08:00
|
|
|
Status error;
|
2017-05-04 18:11:33 +08:00
|
|
|
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
|
2017-12-11 17:33:18 +08:00
|
|
|
TEST_F(MainLoopTest, DetectsEOF) {
|
2017-12-11 18:09:14 +08:00
|
|
|
PseudoTerminal term;
|
2017-12-11 17:33:18 +08:00
|
|
|
ASSERT_TRUE(term.OpenFirstAvailableMaster(O_RDWR, nullptr, 0));
|
|
|
|
ASSERT_TRUE(term.OpenSlave(O_RDWR | O_NOCTTY, nullptr, 0));
|
|
|
|
auto conn = llvm::make_unique<ConnectionFileDescriptor>(
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2017-05-04 18:11:33 +08:00
|
|
|
TEST_F(MainLoopTest, Signal) {
|
|
|
|
MainLoop loop;
|
2017-05-12 12:51:55 +08:00
|
|
|
Status error;
|
2017-05-04 18:11:33 +08:00
|
|
|
|
|
|
|
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
|