// Copyright 2015 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <brillo/message_loops/glib_message_loop.h>
#include <fcntl.h>
#include <unistd.h>
#include <memory>
#include <base/bind.h>
#include <base/location.h>
#include <base/posix/eintr_wrapper.h>
#include <gtest/gtest.h>
#include <brillo/bind_lambda.h>
#include <brillo/message_loops/message_loop.h>
#include <brillo/message_loops/message_loop_utils.h>
using base::Bind;
namespace brillo {
using TaskId = MessageLoop::TaskId;
class GlibMessageLoopTest : public ::testing::Test {
protected:
void SetUp() override {
loop_.reset(new GlibMessageLoop());
EXPECT_TRUE(loop_.get());
}
std::unique_ptr<GlibMessageLoop> loop_;
};
// When you watch a file descriptor for reading, the guaranties are that a
// blocking call to read() on that file descriptor will not block. This should
// include the case when the other end of a pipe is closed or the file is empty.
TEST_F(GlibMessageLoopTest, WatchFileDescriptorTriggersWhenEmpty) {
int fd = HANDLE_EINTR(open("/dev/null", O_RDONLY));
int called = 0;
TaskId task_id = loop_->WatchFileDescriptor(
FROM_HERE, fd, MessageLoop::kWatchRead, true,
Bind([&called] { called++; }));
EXPECT_NE(MessageLoop::kTaskIdNull, task_id);
EXPECT_NE(0, MessageLoopRunMaxIterations(loop_.get(), 10));
EXPECT_LT(2, called);
EXPECT_TRUE(loop_->CancelTask(task_id));
}
// Test that an invalid file descriptor triggers the callback.
TEST_F(GlibMessageLoopTest, WatchFileDescriptorTriggersWhenInvalid) {
int fd = HANDLE_EINTR(open("/dev/zero", O_RDONLY));
int called = 0;
TaskId task_id = loop_->WatchFileDescriptor(
FROM_HERE, fd, MessageLoop::kWatchRead, true,
Bind([&called, fd] {
if (!called)
IGNORE_EINTR(close(fd));
called++;
}));
EXPECT_NE(MessageLoop::kTaskIdNull, task_id);
EXPECT_NE(0, MessageLoopRunMaxIterations(loop_.get(), 10));
EXPECT_LT(2, called);
EXPECT_TRUE(loop_->CancelTask(task_id));
}
} // namespace brillo