// Copyright 2014 The Chromium 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 "base/bind.h" #include "base/message_loop/message_loop.h" #include "base/run_loop.h" #include "base/strings/string_piece.h" #include "device/serial/async_waiter.h" #include "device/serial/buffer.h" #include "device/serial/data_receiver.h" #include "device/serial/data_source_sender.h" #include "device/serial/data_stream.mojom.h" #include "mojo/public/cpp/bindings/interface_ptr.h" #include "testing/gtest/include/gtest/gtest.h" namespace device { class DataSourceTest : public testing::Test { public: enum Event { EVENT_NONE, EVENT_WRITE_BUFFER_READY, EVENT_RECEIVE_COMPLETE, EVENT_ERROR, }; DataSourceTest() : error_(0), seen_connection_error_(false), expected_event_(EVENT_NONE) {} void SetUp() override { message_loop_.reset(new base::MessageLoop); mojo::InterfacePtr source_sender_handle; source_sender_ = mojo::WeakBindToProxy( new DataSourceSender( base::Bind(&DataSourceTest::CanWriteData, base::Unretained(this)), base::Bind(&DataSourceTest::OnError, base::Unretained(this))), &source_sender_handle); receiver_ = new DataReceiver(source_sender_handle.Pass(), 100, kFatalError); } void TearDown() override { write_buffer_.reset(); buffer_.reset(); message_loop_.reset(); } void OnError() { seen_connection_error_ = true; EventReceived(EVENT_ERROR); } void WaitForEvent(Event event) { expected_event_ = event; base::RunLoop run_loop; stop_run_loop_ = run_loop.QuitClosure(); run_loop.Run(); } void EventReceived(Event event) { if (event == expected_event_ && !stop_run_loop_.is_null()) stop_run_loop_.Run(); } bool Receive() { return receiver_->Receive( base::Bind(&DataSourceTest::OnDataReceived, base::Unretained(this)), base::Bind(&DataSourceTest::OnReceiveError, base::Unretained(this))); } void FillWriteBuffer(const base::StringPiece& data, int32_t error) { if (!write_buffer_) { // Run the message loop until CanWriteData is called. WaitForEvent(EVENT_WRITE_BUFFER_READY); } ASSERT_TRUE(write_buffer_); ASSERT_GE(write_buffer_->GetSize(), static_cast(data.size())); memcpy(write_buffer_->GetData(), data.data(), data.size()); if (error) write_buffer_->DoneWithError(static_cast(data.size()), error); else write_buffer_->Done(static_cast(data.size())); write_buffer_.reset(); } void ReceiveAndWait() { ASSERT_TRUE(Receive()); // Run the message loop until OnDataReceived or OnReceiveError is called. WaitForEvent(EVENT_RECEIVE_COMPLETE); } void OnDataReceived(scoped_ptr buffer) { ASSERT_TRUE(buffer); error_ = 0; buffer_ = buffer.Pass(); buffer_contents_ = std::string(buffer_->GetData(), buffer_->GetSize()); EventReceived(EVENT_RECEIVE_COMPLETE); } void OnReceiveError(int32_t error) { buffer_contents_.clear(); error_ = error; EventReceived(EVENT_RECEIVE_COMPLETE); } void CanWriteData(scoped_ptr buffer) { write_buffer_ = buffer.Pass(); EventReceived(EVENT_WRITE_BUFFER_READY); } protected: static const int32_t kFatalError; scoped_ptr message_loop_; base::Closure stop_run_loop_; scoped_refptr source_sender_; scoped_refptr receiver_; scoped_ptr buffer_; std::string buffer_contents_; int32_t error_; scoped_ptr write_buffer_; bool seen_connection_error_; Event expected_event_; private: DISALLOW_COPY_AND_ASSIGN(DataSourceTest); }; const int32_t DataSourceTest::kFatalError = -10; // Test that data is successfully transmitted from the source to the receiver. TEST_F(DataSourceTest, Basic) { ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("a", 0)); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_EQ(0, error_); ASSERT_TRUE(buffer_); EXPECT_EQ("a", buffer_contents_); buffer_->Done(1); ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("b", 0)); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_EQ(0, error_); ASSERT_TRUE(buffer_); EXPECT_EQ("b", buffer_contents_); } // Test that the receiver does not discard any data that is not read by the // client. TEST_F(DataSourceTest, PartialReceive) { ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("ab", 0)); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_EQ(0, error_); ASSERT_TRUE(buffer_); EXPECT_EQ("ab", buffer_contents_); buffer_->Done(1); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_EQ(0, error_); ASSERT_TRUE(buffer_); EXPECT_EQ("b", buffer_contents_); } // Test that an error is correctly reported to the Receive() call immediately // after the data has been read by the client. TEST_F(DataSourceTest, ErrorAndData) { ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("abc", -1)); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); ASSERT_TRUE(buffer_); EXPECT_EQ("abc", buffer_contents_); buffer_->Done(1); EXPECT_EQ(0, error_); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); ASSERT_TRUE(buffer_); EXPECT_EQ("bc", buffer_contents_); buffer_->Done(1); EXPECT_EQ(0, error_); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); ASSERT_TRUE(buffer_); EXPECT_EQ("c", buffer_contents_); buffer_->Done(1); EXPECT_EQ(0, error_); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_EQ(-1, error_); ASSERT_FALSE(write_buffer_); ASSERT_TRUE(Receive()); ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("d", 0)); WaitForEvent(EVENT_RECEIVE_COMPLETE); ASSERT_TRUE(buffer_); EXPECT_EQ("d", buffer_contents_); buffer_->Done(1); EXPECT_EQ(0, error_); ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("e", -2)); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); ASSERT_TRUE(buffer_); EXPECT_EQ("e", buffer_contents_); buffer_->Done(1); EXPECT_EQ(0, error_); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_EQ(-2, error_); } // Test that an error is correctly reported when the source encounters an error // without sending any data. TEST_F(DataSourceTest, ErrorOnly) { ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("", -1)); ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_FALSE(buffer_); EXPECT_EQ(-1, error_); ASSERT_FALSE(write_buffer_); ASSERT_TRUE(Receive()); ASSERT_NO_FATAL_FAILURE(FillWriteBuffer("", -2)); WaitForEvent(EVENT_RECEIVE_COMPLETE); EXPECT_FALSE(buffer_); EXPECT_EQ(-2, error_); ASSERT_FALSE(write_buffer_); } // Test that the source shutting down is correctly reported to the client. TEST_F(DataSourceTest, SourceShutdown) { source_sender_->ShutDown(); source_sender_ = NULL; ASSERT_NO_FATAL_FAILURE(ReceiveAndWait()); EXPECT_FALSE(buffer_); EXPECT_EQ(kFatalError, error_); ASSERT_FALSE(write_buffer_); ASSERT_FALSE(Receive()); } // Test that the receiver shutting down is correctly reported to the source. TEST_F(DataSourceTest, ReceiverShutdown) { Receive(); receiver_ = NULL; EXPECT_EQ(kFatalError, error_); WaitForEvent(EVENT_ERROR); EXPECT_TRUE(seen_connection_error_); } } // namespace device