blob: b725f35e8d0876ddea37afd997041b96080871b4 [file] [log] [blame]
// 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/buffer.h"
#include "device/serial/data_receiver.h"
#include "device/serial/data_source_sender.h"
#include "device/serial/data_stream.mojom.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/mojo/src/mojo/public/cpp/bindings/interface_ptr.h"
#include "third_party/mojo/src/mojo/public/cpp/environment/async_waiter.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<serial::DataSource> source_sender_handle;
mojo::InterfacePtr<serial::DataSourceClient> source_sender_client_handle;
mojo::InterfaceRequest<serial::DataSourceClient>
source_sender_client_request =
mojo::GetProxy(&source_sender_client_handle);
source_sender_ = new DataSourceSender(
mojo::GetProxy(&source_sender_handle),
source_sender_client_handle.Pass(),
base::Bind(&DataSourceTest::CanWriteData, base::Unretained(this)),
base::Bind(&DataSourceTest::OnError, base::Unretained(this)));
receiver_ =
new DataReceiver(source_sender_handle.Pass(),
source_sender_client_request.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<uint32_t>(data.size()));
memcpy(write_buffer_->GetData(), data.data(), data.size());
if (error)
write_buffer_->DoneWithError(static_cast<uint32_t>(data.size()), error);
else
write_buffer_->Done(static_cast<uint32_t>(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<ReadOnlyBuffer> 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<WritableBuffer> buffer) {
write_buffer_ = buffer.Pass();
EventReceived(EVENT_WRITE_BUFFER_READY);
}
protected:
static const int32_t kFatalError;
scoped_ptr<base::MessageLoop> message_loop_;
base::Closure stop_run_loop_;
scoped_refptr<DataSourceSender> source_sender_;
scoped_refptr<DataReceiver> receiver_;
scoped_ptr<ReadOnlyBuffer> buffer_;
std::string buffer_contents_;
int32_t error_;
scoped_ptr<WritableBuffer> 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