• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "udp.h"
18 
19 #include <gtest/gtest.h>
20 
21 #include "socket.h"
22 #include "socket_mock.h"
23 
24 using namespace udp;
25 using namespace udp::internal;
26 
27 // Some possible corner case sequence numbers we want to check.
28 static const uint16_t kTestSequenceNumbers[] = {0x0000, 0x0001, 0x00FF, 0x0100,
29                                                 0x7FFF, 0x8000, 0xFFFF};
30 
31 // Converts |value| to a binary big-endian string.
PacketValue(uint16_t value)32 static std::string PacketValue(uint16_t value) {
33     return std::string{static_cast<char>(value >> 8), static_cast<char>(value)};
34 }
35 
36 // Returns an Error packet.
ErrorPacket(uint16_t sequence,const std::string & message="",char flags=kFlagNone)37 static std::string ErrorPacket(uint16_t sequence, const std::string& message = "",
38                                char flags = kFlagNone) {
39     return std::string{kIdError, flags} + PacketValue(sequence) + message;
40 }
41 
42 // Returns a Query packet with no data.
QueryPacket(uint16_t sequence)43 static std::string QueryPacket(uint16_t sequence) {
44     return std::string{kIdDeviceQuery, kFlagNone} + PacketValue(sequence);
45 }
46 
47 // Returns a Query packet with a 2-byte |new_sequence|.
QueryPacket(uint16_t sequence,uint16_t new_sequence)48 static std::string QueryPacket(uint16_t sequence, uint16_t new_sequence) {
49     return std::string{kIdDeviceQuery, kFlagNone} + PacketValue(sequence) +
50            PacketValue(new_sequence);
51 }
52 
53 // Returns an Init packet with a 2-byte |version| and |max_packet_size|.
InitPacket(uint16_t sequence,uint16_t version,uint16_t max_packet_size)54 static std::string InitPacket(uint16_t sequence, uint16_t version, uint16_t max_packet_size) {
55     return std::string{kIdInitialization, kFlagNone} + PacketValue(sequence) +
56            PacketValue(version) + PacketValue(max_packet_size);
57 }
58 
59 // Returns a Fastboot packet with |data|.
FastbootPacket(uint16_t sequence,const std::string & data="",char flags=kFlagNone)60 static std::string FastbootPacket(uint16_t sequence, const std::string& data = "",
61                                   char flags = kFlagNone) {
62     return std::string{kIdFastboot, flags} + PacketValue(sequence) + data;
63 }
64 
65 // Fixture class to test protocol initialization. Usage is to set up the expected calls to the
66 // SocketMock object then call UdpConnect() and check the result.
67 class UdpConnectTest : public ::testing::Test {
68   public:
UdpConnectTest()69     UdpConnectTest() : mock_socket_(new SocketMock) {}
70 
71     // Run the initialization, return whether it was successful or not. This passes ownership of
72     // the current |mock_socket_| but allocates a new one for re-use.
UdpConnect(std::string * error=nullptr)73     bool UdpConnect(std::string* error = nullptr) {
74         std::string local_error;
75         if (error == nullptr) {
76             error = &local_error;
77         }
78         std::unique_ptr<Transport> transport(Connect(std::move(mock_socket_), error));
79         mock_socket_.reset(new SocketMock);
80         return transport != nullptr && error->empty();
81     }
82 
83   protected:
84     std::unique_ptr<SocketMock> mock_socket_;
85 };
86 
87 // Tests a successful protocol initialization with various starting sequence numbers.
TEST_F(UdpConnectTest,InitializationSuccess)88 TEST_F(UdpConnectTest, InitializationSuccess) {
89     for (uint16_t seq : kTestSequenceNumbers) {
90         mock_socket_->ExpectSend(QueryPacket(0));
91         mock_socket_->AddReceive(QueryPacket(0, seq));
92         mock_socket_->ExpectSend(InitPacket(seq, kProtocolVersion, kHostMaxPacketSize));
93         mock_socket_->AddReceive(InitPacket(seq, kProtocolVersion, 1024));
94 
95         EXPECT_TRUE(UdpConnect());
96     }
97 }
98 
99 // Tests continuation packets during initialization.
TEST_F(UdpConnectTest,InitializationContinuationSuccess)100 TEST_F(UdpConnectTest, InitializationContinuationSuccess) {
101     mock_socket_->ExpectSend(QueryPacket(0));
102     mock_socket_->AddReceive(std::string{kIdDeviceQuery, kFlagContinuation, 0, 0, 0x44});
103     mock_socket_->ExpectSend(std::string{kIdDeviceQuery, kFlagNone, 0, 1});
104     mock_socket_->AddReceive(std::string{kIdDeviceQuery, kFlagNone, 0, 1, 0x55});
105 
106     mock_socket_->ExpectSend(InitPacket(0x4455, kProtocolVersion, kHostMaxPacketSize));
107     mock_socket_->AddReceive(std::string{kIdInitialization, kFlagContinuation, 0x44, 0x55, 0});
108     mock_socket_->ExpectSend(std::string{kIdInitialization, kFlagNone, 0x44, 0x56});
109     mock_socket_->AddReceive(std::string{kIdInitialization, kFlagContinuation, 0x44, 0x56, 1});
110     mock_socket_->ExpectSend(std::string{kIdInitialization, kFlagNone, 0x44, 0x57});
111     mock_socket_->AddReceive(std::string{kIdInitialization, kFlagContinuation, 0x44, 0x57, 2});
112     mock_socket_->ExpectSend(std::string{kIdInitialization, kFlagNone, 0x44, 0x58});
113     mock_socket_->AddReceive(std::string{kIdInitialization, kFlagNone, 0x44, 0x58, 0});
114 
115     EXPECT_TRUE(UdpConnect());
116 }
117 
118 
119 // Tests a mismatched version number; as long as the minimum of the two versions is supported
120 // we should allow the connection.
TEST_F(UdpConnectTest,InitializationVersionMismatch)121 TEST_F(UdpConnectTest, InitializationVersionMismatch) {
122     mock_socket_->ExpectSend(QueryPacket(0));
123     mock_socket_->AddReceive(QueryPacket(0, 0));
124     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
125     mock_socket_->AddReceive(InitPacket(0, 2, 1024));
126 
127     EXPECT_TRUE(UdpConnect());
128 
129     mock_socket_->ExpectSend(QueryPacket(0));
130     mock_socket_->AddReceive(QueryPacket(0, 0));
131     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
132     mock_socket_->AddReceive(InitPacket(0, 0, 1024));
133 
134     EXPECT_FALSE(UdpConnect());
135 }
136 
TEST_F(UdpConnectTest,QueryResponseTimeoutFailure)137 TEST_F(UdpConnectTest, QueryResponseTimeoutFailure) {
138     for (int i = 0; i < kMaxConnectAttempts; ++i) {
139         mock_socket_->ExpectSend(QueryPacket(0));
140         mock_socket_->AddReceiveTimeout();
141     }
142 
143     EXPECT_FALSE(UdpConnect());
144 }
145 
TEST_F(UdpConnectTest,QueryResponseReceiveFailure)146 TEST_F(UdpConnectTest, QueryResponseReceiveFailure) {
147     mock_socket_->ExpectSend(QueryPacket(0));
148     mock_socket_->AddReceiveFailure();
149 
150     EXPECT_FALSE(UdpConnect());
151 }
152 
TEST_F(UdpConnectTest,InitResponseTimeoutFailure)153 TEST_F(UdpConnectTest, InitResponseTimeoutFailure) {
154     mock_socket_->ExpectSend(QueryPacket(0));
155     mock_socket_->AddReceive(QueryPacket(0, 0));
156     for (int i = 0; i < kMaxTransmissionAttempts; ++i) {
157         mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
158         mock_socket_->AddReceiveTimeout();
159     }
160 
161     EXPECT_FALSE(UdpConnect());
162 }
163 
TEST_F(UdpConnectTest,InitResponseReceiveFailure)164 TEST_F(UdpConnectTest, InitResponseReceiveFailure) {
165     mock_socket_->ExpectSend(QueryPacket(0));
166     mock_socket_->AddReceive(QueryPacket(0, 0));
167     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
168     mock_socket_->AddReceiveFailure();
169 
170     EXPECT_FALSE(UdpConnect());
171 }
172 
173 // Tests that we can recover up to the maximum number of allowed retries.
TEST_F(UdpConnectTest,ResponseRecovery)174 TEST_F(UdpConnectTest, ResponseRecovery) {
175     // The device query packet can recover from up to (kMaxConnectAttempts - 1) timeouts.
176     for (int i = 0; i < kMaxConnectAttempts - 1; ++i) {
177         mock_socket_->ExpectSend(QueryPacket(0));
178         mock_socket_->AddReceiveTimeout();
179     }
180     mock_socket_->ExpectSend(QueryPacket(0));
181     mock_socket_->AddReceive(QueryPacket(0, 0));
182 
183     // Subsequent packets try up to (kMaxTransmissionAttempts - 1) times.
184     for (int i = 0; i < kMaxTransmissionAttempts - 1; ++i) {
185         mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
186         mock_socket_->AddReceiveTimeout();
187     }
188     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
189     mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024));
190 
191     EXPECT_TRUE(UdpConnect());
192 }
193 
194 // Tests that the host can handle receiving additional bytes for forward compatibility.
TEST_F(UdpConnectTest,ExtraResponseDataSuccess)195 TEST_F(UdpConnectTest, ExtraResponseDataSuccess) {
196     mock_socket_->ExpectSend(QueryPacket(0));
197     mock_socket_->AddReceive(QueryPacket(0, 0) + "foo");
198     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
199     mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024) + "bar");
200 
201     EXPECT_TRUE(UdpConnect());
202 }
203 
204 // Tests mismatched response sequence numbers. A wrong sequence number is interpreted as a previous
205 // retransmission and just ignored so we should be able to recover.
TEST_F(UdpConnectTest,WrongSequenceRecovery)206 TEST_F(UdpConnectTest, WrongSequenceRecovery) {
207     mock_socket_->ExpectSend(QueryPacket(0));
208     mock_socket_->AddReceive(QueryPacket(1, 0));
209     mock_socket_->AddReceive(QueryPacket(0, 0));
210 
211     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
212     mock_socket_->AddReceive(InitPacket(1, kProtocolVersion, 1024));
213     mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024));
214 
215     EXPECT_TRUE(UdpConnect());
216 }
217 
218 // Tests mismatched response IDs. This should also be interpreted as a retransmission and ignored.
TEST_F(UdpConnectTest,WrongIdRecovery)219 TEST_F(UdpConnectTest, WrongIdRecovery) {
220     mock_socket_->ExpectSend(QueryPacket(0));
221     mock_socket_->AddReceive(FastbootPacket(0));
222     mock_socket_->AddReceive(QueryPacket(0, 0));
223 
224     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
225     mock_socket_->AddReceive(FastbootPacket(0));
226     mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 1024));
227 
228     EXPECT_TRUE(UdpConnect());
229 }
230 
231 // Tests an invalid query response. Query responses must have at least 2 bytes of data.
TEST_F(UdpConnectTest,InvalidQueryResponseFailure)232 TEST_F(UdpConnectTest, InvalidQueryResponseFailure) {
233     std::string error;
234 
235     mock_socket_->ExpectSend(QueryPacket(0));
236     mock_socket_->AddReceive(QueryPacket(0));
237 
238     EXPECT_FALSE(UdpConnect(&error));
239     EXPECT_EQ("invalid query response from target", error);
240 
241     mock_socket_->ExpectSend(QueryPacket(0));
242     mock_socket_->AddReceive(QueryPacket(0) + std::string{0x00});
243 
244     EXPECT_FALSE(UdpConnect(&error));
245     EXPECT_EQ("invalid query response from target", error);
246 }
247 
248 // Tests an invalid initialization response. Max packet size must be at least 512 bytes.
TEST_F(UdpConnectTest,InvalidInitResponseFailure)249 TEST_F(UdpConnectTest, InvalidInitResponseFailure) {
250     std::string error;
251 
252     mock_socket_->ExpectSend(QueryPacket(0));
253     mock_socket_->AddReceive(QueryPacket(0, 0));
254     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
255     mock_socket_->AddReceive(InitPacket(0, kProtocolVersion, 511));
256 
257     EXPECT_FALSE(UdpConnect(&error));
258     EXPECT_EQ("target reported invalid packet size 511", error);
259 
260     mock_socket_->ExpectSend(QueryPacket(0));
261     mock_socket_->AddReceive(QueryPacket(0, 0));
262     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
263     mock_socket_->AddReceive(InitPacket(0, 0, 1024));
264 
265     EXPECT_FALSE(UdpConnect(&error));
266     EXPECT_EQ("target reported invalid protocol version 0", error);
267 }
268 
TEST_F(UdpConnectTest,ErrorResponseFailure)269 TEST_F(UdpConnectTest, ErrorResponseFailure) {
270     std::string error;
271 
272     mock_socket_->ExpectSend(QueryPacket(0));
273     mock_socket_->AddReceive(ErrorPacket(0, "error1"));
274 
275     EXPECT_FALSE(UdpConnect(&error));
276     EXPECT_NE(std::string::npos, error.find("error1"));
277 
278     mock_socket_->ExpectSend(QueryPacket(0));
279     mock_socket_->AddReceive(QueryPacket(0, 0));
280     mock_socket_->ExpectSend(InitPacket(0, kProtocolVersion, kHostMaxPacketSize));
281     mock_socket_->AddReceive(ErrorPacket(0, "error2"));
282 
283     EXPECT_FALSE(UdpConnect(&error));
284     EXPECT_NE(std::string::npos, error.find("error2"));
285 }
286 
287 // Tests an error response with continuation flag.
TEST_F(UdpConnectTest,ErrorContinuationFailure)288 TEST_F(UdpConnectTest, ErrorContinuationFailure) {
289     std::string error;
290 
291     mock_socket_->ExpectSend(QueryPacket(0));
292     mock_socket_->AddReceive(ErrorPacket(0, "error1", kFlagContinuation));
293     mock_socket_->ExpectSend(ErrorPacket(1));
294     mock_socket_->AddReceive(ErrorPacket(1, " ", kFlagContinuation));
295     mock_socket_->ExpectSend(ErrorPacket(2));
296     mock_socket_->AddReceive(ErrorPacket(2, "error2"));
297 
298     EXPECT_FALSE(UdpConnect(&error));
299     EXPECT_NE(std::string::npos, error.find("error1 error2"));
300 }
301 
302 // Fixture class to test UDP Transport read/write functionality.
303 class UdpTest : public ::testing::Test {
304   public:
SetUp()305     void SetUp() override {
306         // Create |transport_| starting at sequence 0 with 512 byte max packet size. Tests can call
307         // InitializeTransport() again to change settings.
308         ASSERT_TRUE(InitializeTransport(0, 512));
309     }
310 
311     // Sets up |mock_socket_| to correctly initialize the protocol and creates |transport_|. This
312     // can be called multiple times in a test if needed.
InitializeTransport(uint16_t starting_sequence,int device_max_packet_size=512)313     bool InitializeTransport(uint16_t starting_sequence, int device_max_packet_size = 512) {
314         mock_socket_ = new SocketMock;
315         mock_socket_->ExpectSend(QueryPacket(0));
316         mock_socket_->AddReceive(QueryPacket(0, starting_sequence));
317         mock_socket_->ExpectSend(
318                 InitPacket(starting_sequence, kProtocolVersion, kHostMaxPacketSize));
319         mock_socket_->AddReceive(
320                 InitPacket(starting_sequence, kProtocolVersion, device_max_packet_size));
321 
322         std::string error;
323         transport_ = Connect(std::unique_ptr<Socket>(mock_socket_), &error);
324         return transport_ != nullptr && error.empty();
325     }
326 
327     // Writes |message| to |transport_|, returns true on success.
Write(const std::string & message)328     bool Write(const std::string& message) {
329         return transport_->Write(message.data(), message.length()) ==
330                 static_cast<ssize_t>(message.length());
331     }
332 
333     // Reads from |transport_|, returns true if it matches |message|.
Read(const std::string & message)334     bool Read(const std::string& message) {
335         std::string buffer(message.length(), '\0');
336         return transport_->Read(&buffer[0], buffer.length()) ==
337                 static_cast<ssize_t>(message.length()) && buffer == message;
338     }
339 
340   protected:
341     // |mock_socket_| is a raw pointer here because we transfer ownership to |transport_| but we
342     // need to retain a pointer to set send and receive expectations.
343     SocketMock* mock_socket_ = nullptr;
344     std::unique_ptr<Transport> transport_;
345 };
346 
347 // Tests sequence behavior with various starting sequence numbers.
TEST_F(UdpTest,SequenceIncrementCheck)348 TEST_F(UdpTest, SequenceIncrementCheck) {
349     for (uint16_t seq : kTestSequenceNumbers) {
350         ASSERT_TRUE(InitializeTransport(seq));
351 
352         for (int i = 0; i < 10; ++i) {
353             mock_socket_->ExpectSend(FastbootPacket(++seq, "foo"));
354             mock_socket_->AddReceive(FastbootPacket(seq, ""));
355             mock_socket_->ExpectSend(FastbootPacket(++seq, ""));
356             mock_socket_->AddReceive(FastbootPacket(seq, "bar"));
357 
358             EXPECT_TRUE(Write("foo"));
359             EXPECT_TRUE(Read("bar"));
360         }
361     }
362 }
363 
364 // Tests sending and receiving a few small packets.
TEST_F(UdpTest,ReadAndWriteSmallPackets)365 TEST_F(UdpTest, ReadAndWriteSmallPackets) {
366     mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
367     mock_socket_->AddReceive(FastbootPacket(1, ""));
368     mock_socket_->ExpectSend(FastbootPacket(2, ""));
369     mock_socket_->AddReceive(FastbootPacket(2, "bar"));
370 
371     EXPECT_TRUE(Write("foo"));
372     EXPECT_TRUE(Read("bar"));
373 
374     mock_socket_->ExpectSend(FastbootPacket(3, "12345 67890"));
375     mock_socket_->AddReceive(FastbootPacket(3));
376     mock_socket_->ExpectSend(FastbootPacket(4, "\x01\x02\x03\x04\x05"));
377     mock_socket_->AddReceive(FastbootPacket(4));
378 
379     EXPECT_TRUE(Write("12345 67890"));
380     EXPECT_TRUE(Write("\x01\x02\x03\x04\x05"));
381 
382     // Reads are done by sending empty packets.
383     mock_socket_->ExpectSend(FastbootPacket(5));
384     mock_socket_->AddReceive(FastbootPacket(5, "foo bar baz"));
385     mock_socket_->ExpectSend(FastbootPacket(6));
386     mock_socket_->AddReceive(FastbootPacket(6, "\x01\x02\x03\x04\x05"));
387 
388     EXPECT_TRUE(Read("foo bar baz"));
389     EXPECT_TRUE(Read("\x01\x02\x03\x04\x05"));
390 }
391 
TEST_F(UdpTest,ResponseTimeoutFailure)392 TEST_F(UdpTest, ResponseTimeoutFailure) {
393     for (int i = 0; i < kMaxTransmissionAttempts; ++i) {
394         mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
395         mock_socket_->AddReceiveTimeout();
396     }
397 
398     EXPECT_FALSE(Write("foo"));
399 }
400 
TEST_F(UdpTest,ResponseReceiveFailure)401 TEST_F(UdpTest, ResponseReceiveFailure) {
402     mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
403     mock_socket_->AddReceiveFailure();
404 
405     EXPECT_FALSE(Write("foo"));
406 }
407 
TEST_F(UdpTest,ResponseTimeoutRecovery)408 TEST_F(UdpTest, ResponseTimeoutRecovery) {
409     for (int i = 0; i < kMaxTransmissionAttempts - 1; ++i) {
410         mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
411         mock_socket_->AddReceiveTimeout();
412     }
413     mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
414     mock_socket_->AddReceive(FastbootPacket(1, ""));
415 
416     EXPECT_TRUE(Write("foo"));
417 }
418 
419 // Tests continuation packets for various max packet sizes.
420 // The important part of this test is that regardless of what kind of packet fragmentation happens
421 // at the socket layer, a single call to Transport::Read() and Transport::Write() is all the
422 // fastboot code needs to do.
TEST_F(UdpTest,ContinuationPackets)423 TEST_F(UdpTest, ContinuationPackets) {
424     for (uint16_t max_packet_size : {512, 1024, 1200}) {
425         ASSERT_TRUE(InitializeTransport(0, max_packet_size));
426 
427         // Initialize the data we want to send. Use (size - 4) to leave room for the header.
428         size_t max_data_size = max_packet_size - 4;
429         std::string data(max_data_size * 3, '\0');
430         for (size_t i = 0; i < data.length(); ++i) {
431             data[i] = i;
432         }
433         std::string chunks[] = {data.substr(0, max_data_size),
434                                 data.substr(max_data_size, max_data_size),
435                                 data.substr(max_data_size * 2, max_data_size)};
436 
437         // Write data: split into 3 UDP packets, each of which will be ACKed.
438         mock_socket_->ExpectSend(FastbootPacket(1, chunks[0], kFlagContinuation));
439         mock_socket_->AddReceive(FastbootPacket(1));
440         mock_socket_->ExpectSend(FastbootPacket(2, chunks[1], kFlagContinuation));
441         mock_socket_->AddReceive(FastbootPacket(2));
442         mock_socket_->ExpectSend(FastbootPacket(3, chunks[2]));
443         mock_socket_->AddReceive(FastbootPacket(3));
444         EXPECT_TRUE(Write(data));
445 
446         // Same thing for reading the data.
447         mock_socket_->ExpectSend(FastbootPacket(4));
448         mock_socket_->AddReceive(FastbootPacket(4, chunks[0], kFlagContinuation));
449         mock_socket_->ExpectSend(FastbootPacket(5));
450         mock_socket_->AddReceive(FastbootPacket(5, chunks[1], kFlagContinuation));
451         mock_socket_->ExpectSend(FastbootPacket(6));
452         mock_socket_->AddReceive(FastbootPacket(6, chunks[2]));
453         EXPECT_TRUE(Read(data));
454     }
455 }
456 
457 // Tests that the continuation bit is respected even if the packet isn't max size.
TEST_F(UdpTest,SmallContinuationPackets)458 TEST_F(UdpTest, SmallContinuationPackets) {
459     mock_socket_->ExpectSend(FastbootPacket(1));
460     mock_socket_->AddReceive(FastbootPacket(1, "foo", kFlagContinuation));
461     mock_socket_->ExpectSend(FastbootPacket(2));
462     mock_socket_->AddReceive(FastbootPacket(2, "bar"));
463 
464     EXPECT_TRUE(Read("foobar"));
465 }
466 
467 // Tests receiving an error packet mid-continuation.
TEST_F(UdpTest,ContinuationPacketError)468 TEST_F(UdpTest, ContinuationPacketError) {
469     mock_socket_->ExpectSend(FastbootPacket(1));
470     mock_socket_->AddReceive(FastbootPacket(1, "foo", kFlagContinuation));
471     mock_socket_->ExpectSend(FastbootPacket(2));
472     mock_socket_->AddReceive(ErrorPacket(2, "test error"));
473 
474     EXPECT_FALSE(Read("foo"));
475 }
476 
477 // Tests timeout during a continuation sequence.
TEST_F(UdpTest,ContinuationTimeoutRecovery)478 TEST_F(UdpTest, ContinuationTimeoutRecovery) {
479     mock_socket_->ExpectSend(FastbootPacket(1));
480     mock_socket_->AddReceive(FastbootPacket(1, "foo", kFlagContinuation));
481     mock_socket_->ExpectSend(FastbootPacket(2));
482     mock_socket_->AddReceiveTimeout();
483     mock_socket_->ExpectSend(FastbootPacket(2));
484     mock_socket_->AddReceive(FastbootPacket(2, "bar"));
485 
486     EXPECT_TRUE(Read("foobar"));
487 }
488 
489 // Tests read overflow returns -1 to indicate the failure.
TEST_F(UdpTest,MultipleReadPacket)490 TEST_F(UdpTest, MultipleReadPacket) {
491     mock_socket_->ExpectSend(FastbootPacket(1));
492     mock_socket_->AddReceive(FastbootPacket(1, "foobarbaz"));
493 
494     char buffer[3];
495     EXPECT_EQ(-1, transport_->Read(buffer, 3));
496 }
497 
498 // Tests that packets arriving out-of-order are ignored.
TEST_F(UdpTest,IgnoreOutOfOrderPackets)499 TEST_F(UdpTest, IgnoreOutOfOrderPackets) {
500     mock_socket_->ExpectSend(FastbootPacket(1));
501     mock_socket_->AddReceive(FastbootPacket(0, "sequence too low"));
502     mock_socket_->AddReceive(FastbootPacket(2, "sequence too high"));
503     mock_socket_->AddReceive(QueryPacket(1));
504     mock_socket_->AddReceive(FastbootPacket(1, "correct"));
505 
506     EXPECT_TRUE(Read("correct"));
507 }
508 
509 // Tests that an error response with the correct sequence number causes immediate failure.
TEST_F(UdpTest,ErrorResponse)510 TEST_F(UdpTest, ErrorResponse) {
511     // Error packets with the wrong sequence number should be ignored like any other packet.
512     mock_socket_->ExpectSend(FastbootPacket(1, "foo"));
513     mock_socket_->AddReceive(ErrorPacket(0, "ignored error"));
514     mock_socket_->AddReceive(FastbootPacket(1));
515 
516     EXPECT_TRUE(Write("foo"));
517 
518     // Error packets with the correct sequence should abort immediately without retransmission.
519     mock_socket_->ExpectSend(FastbootPacket(2, "foo"));
520     mock_socket_->AddReceive(ErrorPacket(2, "test error"));
521 
522     EXPECT_FALSE(Write("foo"));
523 }
524 
525 // Tests that attempting to use a closed transport returns -1 without making any socket calls.
TEST_F(UdpTest,CloseTransport)526 TEST_F(UdpTest, CloseTransport) {
527     char buffer[32];
528     EXPECT_EQ(0, transport_->Close());
529     EXPECT_EQ(-1, transport_->Write("foo", 3));
530     EXPECT_EQ(-1, transport_->Read(buffer, sizeof(buffer)));
531 }
532