532 lines
21 KiB
C++
532 lines
21 KiB
C++
|
/*
|
||
|
* Copyright (C) 2015 The Android Open Source Project
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
#include "udp.h"
|
||
|
|
||
|
#include <gtest/gtest.h>
|
||
|
|
||
|
#include "socket.h"
|
||
|
#include "socket_mock.h"
|
||
|
|
||
|
using namespace udp;
|
||
|
using namespace udp::internal;
|
||
|
|
||
|
// Some possible corner case sequence numbers we want to check.
|
||
|
static const uint16_t kTestSequenceNumbers[] = {0x0000, 0x0001, 0x00FF, 0x0100,
|
||
|
0x7FFF, 0x8000, 0xFFFF};
|
||
|
|
||
|
// Converts |value| to a binary big-endian string.
|
||
|
static std::string PacketValue(uint16_t value) {
|
||
|
return std::string{static_cast<char>(value >> 8), static_cast<char>(value)};
|
||
|
}
|
||
|
|
||
|
// Returns an Error packet.
|
||
|
static std::string ErrorPacket(uint16_t sequence, const std::string& message = "",
|
||
|
char flags = kFlagNone) {
|
||
|
return std::string{kIdError, flags} + PacketValue(sequence) + message;
|
||
|
}
|
||
|
|
||
|
// Returns a Query packet with no data.
|
||
|
static std::string QueryPacket(uint16_t sequence) {
|
||
|
return std::string{kIdDeviceQuery, kFlagNone} + PacketValue(sequence);
|
||
|
}
|
||
|
|
||
|
// Returns a Query packet with a 2-byte |new_sequence|.
|
||
|
static std::string QueryPacket(uint16_t sequence, uint16_t new_sequence) {
|
||
|
return std::string{kIdDeviceQuery, kFlagNone} + PacketValue(sequence) +
|
||
|
PacketValue(new_sequence);
|
||
|
}
|
||
|
|
||
|
// Returns an Init packet with a 2-byte |version| and |max_packet_size|.
|
||
|
static std::string InitPacket(uint16_t sequence, uint16_t version, uint16_t max_packet_size) {
|
||
|
return std::string{kIdInitialization, kFlagNone} + PacketValue(sequence) +
|
||
|
PacketValue(version) + PacketValue(max_packet_size);
|
||
|
}
|
||
|
|
||
|
// Returns a Fastboot packet with |data|.
|
||
|
static std::string FastbootPacket(uint16_t sequence, const std::string& data = "",
|
||
|
char flags = kFlagNone) {
|
||
|
return std::string{kIdFastboot, flags} + PacketValue(sequence) + data;
|
||
|
}
|
||
|
|
||
|
// Fixture class to test protocol initialization. Usage is to set up the expected calls to the
|
||
|
// SocketMock object then call UdpConnect() and check the result.
|
||
|
class UdpConnectTest : public ::testing::Test {
|
||
|
public:
|
||
|
UdpConnectTest() : mock_socket_(new SocketMock) {}
|
||
|
|
||
|
// Run the initialization, return whether it was successful or not. This passes ownership of
|
||
|
// the current |mock_socket_| but allocates a new one for re-use.
|
||
|
bool UdpConnect(std::string* error = nullptr) {
|
||
|
std::string local_error;
|
||
|
if (error == nullptr) {
|
||
|
error = &local_error;
|
||
|
}
|
||
|
std::unique_ptr<Transport> transport(Connect(std::move(mock_socket_), error));
|
||
|
mock_socket_.reset(new SocketMock);
|
||
|
return transport != nullptr && error->empty();
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
std::unique_ptr<SocketMock> mock_socket_;
|
||
|
};
|
||
|
|
||
|
// Tests a successful protocol initialization with various starting sequence numbers.
|
||
|
TEST_F(UdpConnectTest, InitializationSuccess) {
|
||
|
for (uint16_t seq : kTestSequenceNumbers) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, seq));
|
||
|
mock_socket_->ExpectSend(InitPacket(seq, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(seq, kProtocolVersion, 1024));
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Tests continuation packets during initialization.
|
||
|
TEST_F(UdpConnectTest, InitializationContinuationSuccess) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(std::string{kIdDeviceQuery, kFlagContinuation, 0, 0, 0x44});
|
||
|
mock_socket_->ExpectSend(std::string{kIdDeviceQuery, kFlagNone, 0, 1});
|
||
|
mock_socket_->AddReceive(std::string{kIdDeviceQuery, kFlagNone, 0, 1, 0x55});
|
||
|
|
||
|
mock_socket_->ExpectSend(InitPacket(0x4455, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(std::string{kIdInitialization, kFlagContinuation, 0x44, 0x55, 0});
|
||
|
mock_socket_->ExpectSend(std::string{kIdInitialization, kFlagNone, 0x44, 0x56});
|
||
|
mock_socket_->AddReceive(std::string{kIdInitialization, kFlagContinuation, 0x44, 0x56, 1});
|
||
|
mock_socket_->ExpectSend(std::string{kIdInitialization, kFlagNone, 0x44, 0x57});
|
||
|
mock_socket_->AddReceive(std::string{kIdInitialization, kFlagContinuation, 0x44, 0x57, 2});
|
||
|
mock_socket_->ExpectSend(std::string{kIdInitialization, kFlagNone, 0x44, 0x58});
|
||
|
mock_socket_->AddReceive(std::string{kIdInitialization, kFlagNone, 0x44, 0x58, 0});
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
|
||
|
// Tests a mismatched version number; as long as the minimum of the two versions is supported
|
||
|
// we should allow the connection.
|
||
|
TEST_F(UdpConnectTest, InitializationVersionMismatch) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(0, 2, 1024));
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(0, 0, 1024));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpConnectTest, QueryResponseTimeoutFailure) {
|
||
|
for (int i = 0; i < kMaxConnectAttempts; ++i) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
}
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpConnectTest, QueryResponseReceiveFailure) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceiveFailure();
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpConnectTest, InitResponseTimeoutFailure) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
for (int i = 0; i < kMaxTransmissionAttempts; ++i) {
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
}
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpConnectTest, InitResponseReceiveFailure) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceiveFailure();
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
// Tests that we can recover up to the maximum number of allowed retries.
|
||
|
TEST_F(UdpConnectTest, ResponseRecovery) {
|
||
|
// The device query packet can recover from up to (kMaxConnectAttempts - 1) timeouts.
|
||
|
for (int i = 0; i < kMaxConnectAttempts - 1; ++i) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
}
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
|
||
|
// Subsequent packets try up to (kMaxTransmissionAttempts - 1) times.
|
||
|
for (int i = 0; i < kMaxTransmissionAttempts - 1; ++i) {
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
}
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024));
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
// Tests that the host can handle receiving additional bytes for forward compatibility.
|
||
|
TEST_F(UdpConnectTest, ExtraResponseDataSuccess) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0) + "foo");
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024) + "bar");
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
// Tests mismatched response sequence numbers. A wrong sequence number is interpreted as a previous
|
||
|
// retransmission and just ignored so we should be able to recover.
|
||
|
TEST_F(UdpConnectTest, WrongSequenceRecovery) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(1, 0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(1, kProtocolVersion, 1024));
|
||
|
mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024));
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
// Tests mismatched response IDs. This should also be interpreted as a retransmission and ignored.
|
||
|
TEST_F(UdpConnectTest, WrongIdRecovery) {
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(FastbootPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(FastbootPacket(0));
|
||
|
mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024));
|
||
|
|
||
|
EXPECT_TRUE(UdpConnect());
|
||
|
}
|
||
|
|
||
|
// Tests an invalid query response. Query responses must have at least 2 bytes of data.
|
||
|
TEST_F(UdpConnectTest, InvalidQueryResponseFailure) {
|
||
|
std::string error;
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_EQ("invalid query response from target", error);
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0) + std::string{0x00});
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_EQ("invalid query response from target", error);
|
||
|
}
|
||
|
|
||
|
// Tests an invalid initialization response. Max packet size must be at least 512 bytes.
|
||
|
TEST_F(UdpConnectTest, InvalidInitResponseFailure) {
|
||
|
std::string error;
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 511));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_EQ("target reported invalid packet size 511", error);
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(InitPacket(0, 0, 1024));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_EQ("target reported invalid protocol version 0", error);
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpConnectTest, ErrorResponseFailure) {
|
||
|
std::string error;
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(ErrorPacket(0, "error1"));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_NE(std::string::npos, error.find("error1"));
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, 0));
|
||
|
mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(ErrorPacket(0, "error2"));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_NE(std::string::npos, error.find("error2"));
|
||
|
}
|
||
|
|
||
|
// Tests an error response with continuation flag.
|
||
|
TEST_F(UdpConnectTest, ErrorContinuationFailure) {
|
||
|
std::string error;
|
||
|
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(ErrorPacket(0, "error1", kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(ErrorPacket(1));
|
||
|
mock_socket_->AddReceive(ErrorPacket(1, " ", kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(ErrorPacket(2));
|
||
|
mock_socket_->AddReceive(ErrorPacket(2, "error2"));
|
||
|
|
||
|
EXPECT_FALSE(UdpConnect(&error));
|
||
|
EXPECT_NE(std::string::npos, error.find("error1 error2"));
|
||
|
}
|
||
|
|
||
|
// Fixture class to test UDP Transport read/write functionality.
|
||
|
class UdpTest : public ::testing::Test {
|
||
|
public:
|
||
|
void SetUp() override {
|
||
|
// Create |transport_| starting at sequence 0 with 512 byte max packet size. Tests can call
|
||
|
// InitializeTransport() again to change settings.
|
||
|
ASSERT_TRUE(InitializeTransport(0, 512));
|
||
|
}
|
||
|
|
||
|
// Sets up |mock_socket_| to correctly initialize the protocol and creates |transport_|. This
|
||
|
// can be called multiple times in a test if needed.
|
||
|
bool InitializeTransport(uint16_t starting_sequence, int device_max_packet_size = 512) {
|
||
|
mock_socket_ = new SocketMock;
|
||
|
mock_socket_->ExpectSend(QueryPacket(0));
|
||
|
mock_socket_->AddReceive(QueryPacket(0, starting_sequence));
|
||
|
mock_socket_->ExpectSend(
|
||
|
InitPacket(starting_sequence, kProtocolVersion, kHostMaxPacketSize));
|
||
|
mock_socket_->AddReceive(
|
||
|
InitPacket(starting_sequence, kProtocolVersion, device_max_packet_size));
|
||
|
|
||
|
std::string error;
|
||
|
transport_ = Connect(std::unique_ptr<Socket>(mock_socket_), &error);
|
||
|
return transport_ != nullptr && error.empty();
|
||
|
}
|
||
|
|
||
|
// Writes |message| to |transport_|, returns true on success.
|
||
|
bool Write(const std::string& message) {
|
||
|
return transport_->Write(message.data(), message.length()) ==
|
||
|
static_cast<ssize_t>(message.length());
|
||
|
}
|
||
|
|
||
|
// Reads from |transport_|, returns true if it matches |message|.
|
||
|
bool Read(const std::string& message) {
|
||
|
std::string buffer(message.length(), '\0');
|
||
|
return transport_->Read(&buffer[0], buffer.length()) ==
|
||
|
static_cast<ssize_t>(message.length()) && buffer == message;
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
// |mock_socket_| is a raw pointer here because we transfer ownership to |transport_| but we
|
||
|
// need to retain a pointer to set send and receive expectations.
|
||
|
SocketMock* mock_socket_ = nullptr;
|
||
|
std::unique_ptr<Transport> transport_;
|
||
|
};
|
||
|
|
||
|
// Tests sequence behavior with various starting sequence numbers.
|
||
|
TEST_F(UdpTest, SequenceIncrementCheck) {
|
||
|
for (uint16_t seq : kTestSequenceNumbers) {
|
||
|
ASSERT_TRUE(InitializeTransport(seq));
|
||
|
|
||
|
for (int i = 0; i < 10; ++i) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(++seq, "foo"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(seq, ""));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(++seq, ""));
|
||
|
mock_socket_->AddReceive(FastbootPacket(seq, "bar"));
|
||
|
|
||
|
EXPECT_TRUE(Write("foo"));
|
||
|
EXPECT_TRUE(Read("bar"));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Tests sending and receiving a few small packets.
|
||
|
TEST_F(UdpTest, ReadAndWriteSmallPackets) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, ""));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2, ""));
|
||
|
mock_socket_->AddReceive(FastbootPacket(2, "bar"));
|
||
|
|
||
|
EXPECT_TRUE(Write("foo"));
|
||
|
EXPECT_TRUE(Read("bar"));
|
||
|
|
||
|
mock_socket_->ExpectSend(FastbootPacket(3, "12345 67890"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(3));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(4, "\x01\x02\x03\x04\x05"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(4));
|
||
|
|
||
|
EXPECT_TRUE(Write("12345 67890"));
|
||
|
EXPECT_TRUE(Write("\x01\x02\x03\x04\x05"));
|
||
|
|
||
|
// Reads are done by sending empty packets.
|
||
|
mock_socket_->ExpectSend(FastbootPacket(5));
|
||
|
mock_socket_->AddReceive(FastbootPacket(5, "foo bar baz"));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(6));
|
||
|
mock_socket_->AddReceive(FastbootPacket(6, "\x01\x02\x03\x04\x05"));
|
||
|
|
||
|
EXPECT_TRUE(Read("foo bar baz"));
|
||
|
EXPECT_TRUE(Read("\x01\x02\x03\x04\x05"));
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpTest, ResponseTimeoutFailure) {
|
||
|
for (int i = 0; i < kMaxTransmissionAttempts; ++i) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
}
|
||
|
|
||
|
EXPECT_FALSE(Write("foo"));
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpTest, ResponseReceiveFailure) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
|
||
|
mock_socket_->AddReceiveFailure();
|
||
|
|
||
|
EXPECT_FALSE(Write("foo"));
|
||
|
}
|
||
|
|
||
|
TEST_F(UdpTest, ResponseTimeoutRecovery) {
|
||
|
for (int i = 0; i < kMaxTransmissionAttempts - 1; ++i) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
}
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, ""));
|
||
|
|
||
|
EXPECT_TRUE(Write("foo"));
|
||
|
}
|
||
|
|
||
|
// Tests continuation packets for various max packet sizes.
|
||
|
// The important part of this test is that regardless of what kind of packet fragmentation happens
|
||
|
// at the socket layer, a single call to Transport::Read() and Transport::Write() is all the
|
||
|
// fastboot code needs to do.
|
||
|
TEST_F(UdpTest, ContinuationPackets) {
|
||
|
for (uint16_t max_packet_size : {512, 1024, 1200}) {
|
||
|
ASSERT_TRUE(InitializeTransport(0, max_packet_size));
|
||
|
|
||
|
// Initialize the data we want to send. Use (size - 4) to leave room for the header.
|
||
|
size_t max_data_size = max_packet_size - 4;
|
||
|
std::string data(max_data_size * 3, '\0');
|
||
|
for (size_t i = 0; i < data.length(); ++i) {
|
||
|
data[i] = i;
|
||
|
}
|
||
|
std::string chunks[] = {data.substr(0, max_data_size),
|
||
|
data.substr(max_data_size, max_data_size),
|
||
|
data.substr(max_data_size * 2, max_data_size)};
|
||
|
|
||
|
// Write data: split into 3 UDP packets, each of which will be ACKed.
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, chunks[0], kFlagContinuation));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2, chunks[1], kFlagContinuation));
|
||
|
mock_socket_->AddReceive(FastbootPacket(2));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(3, chunks[2]));
|
||
|
mock_socket_->AddReceive(FastbootPacket(3));
|
||
|
EXPECT_TRUE(Write(data));
|
||
|
|
||
|
// Same thing for reading the data.
|
||
|
mock_socket_->ExpectSend(FastbootPacket(4));
|
||
|
mock_socket_->AddReceive(FastbootPacket(4, chunks[0], kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(5));
|
||
|
mock_socket_->AddReceive(FastbootPacket(5, chunks[1], kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(6));
|
||
|
mock_socket_->AddReceive(FastbootPacket(6, chunks[2]));
|
||
|
EXPECT_TRUE(Read(data));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Tests that the continuation bit is respected even if the packet isn't max size.
|
||
|
TEST_F(UdpTest, SmallContinuationPackets) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, "foo", kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2));
|
||
|
mock_socket_->AddReceive(FastbootPacket(2, "bar"));
|
||
|
|
||
|
EXPECT_TRUE(Read("foobar"));
|
||
|
}
|
||
|
|
||
|
// Tests receiving an error packet mid-continuation.
|
||
|
TEST_F(UdpTest, ContinuationPacketError) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, "foo", kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2));
|
||
|
mock_socket_->AddReceive(ErrorPacket(2, "test error"));
|
||
|
|
||
|
EXPECT_FALSE(Read("foo"));
|
||
|
}
|
||
|
|
||
|
// Tests timeout during a continuation sequence.
|
||
|
TEST_F(UdpTest, ContinuationTimeoutRecovery) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, "foo", kFlagContinuation));
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2));
|
||
|
mock_socket_->AddReceiveTimeout();
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2));
|
||
|
mock_socket_->AddReceive(FastbootPacket(2, "bar"));
|
||
|
|
||
|
EXPECT_TRUE(Read("foobar"));
|
||
|
}
|
||
|
|
||
|
// Tests read overflow returns -1 to indicate the failure.
|
||
|
TEST_F(UdpTest, MultipleReadPacket) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, "foobarbaz"));
|
||
|
|
||
|
char buffer[3];
|
||
|
EXPECT_EQ(-1, transport_->Read(buffer, 3));
|
||
|
}
|
||
|
|
||
|
// Tests that packets arriving out-of-order are ignored.
|
||
|
TEST_F(UdpTest, IgnoreOutOfOrderPackets) {
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1));
|
||
|
mock_socket_->AddReceive(FastbootPacket(0, "sequence too low"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(2, "sequence too high"));
|
||
|
mock_socket_->AddReceive(QueryPacket(1));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1, "correct"));
|
||
|
|
||
|
EXPECT_TRUE(Read("correct"));
|
||
|
}
|
||
|
|
||
|
// Tests that an error response with the correct sequence number causes immediate failure.
|
||
|
TEST_F(UdpTest, ErrorResponse) {
|
||
|
// Error packets with the wrong sequence number should be ignored like any other packet.
|
||
|
mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
|
||
|
mock_socket_->AddReceive(ErrorPacket(0, "ignored error"));
|
||
|
mock_socket_->AddReceive(FastbootPacket(1));
|
||
|
|
||
|
EXPECT_TRUE(Write("foo"));
|
||
|
|
||
|
// Error packets with the correct sequence should abort immediately without retransmission.
|
||
|
mock_socket_->ExpectSend(FastbootPacket(2, "foo"));
|
||
|
mock_socket_->AddReceive(ErrorPacket(2, "test error"));
|
||
|
|
||
|
EXPECT_FALSE(Write("foo"));
|
||
|
}
|
||
|
|
||
|
// Tests that attempting to use a closed transport returns -1 without making any socket calls.
|
||
|
TEST_F(UdpTest, CloseTransport) {
|
||
|
char buffer[32];
|
||
|
EXPECT_EQ(0, transport_->Close());
|
||
|
EXPECT_EQ(-1, transport_->Write("foo", 3));
|
||
|
EXPECT_EQ(-1, transport_->Read(buffer, sizeof(buffer)));
|
||
|
}
|