OLD | NEW |
| (Empty) |
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "net/spdy/spdy_protocol.h" | |
6 | |
7 #include "base/memory/scoped_ptr.h" | |
8 #include "net/spdy/spdy_bitmasks.h" | |
9 #include "net/spdy/spdy_framer.h" | |
10 #include "testing/platform_test.h" | |
11 | |
12 using spdy::CONTROL_FLAG_FIN; | |
13 using spdy::CONTROL_FLAG_NONE; | |
14 using spdy::GOAWAY; | |
15 using spdy::HEADERS; | |
16 using spdy::NOOP; | |
17 using spdy::NUM_CONTROL_FRAME_TYPES; | |
18 using spdy::PING; | |
19 using spdy::RST_STREAM; | |
20 using spdy::SETTINGS; | |
21 using spdy::SYN_REPLY; | |
22 using spdy::SYN_STREAM; | |
23 using spdy::WINDOW_UPDATE; | |
24 using spdy::FlagsAndLength; | |
25 using spdy::SpdyControlFrame; | |
26 using spdy::SpdyControlType; | |
27 using spdy::SpdyDataFrame; | |
28 using spdy::SpdyFrame; | |
29 using spdy::SpdyFramer; | |
30 using spdy::SpdyHeaderBlock; | |
31 using spdy::SpdyHeadersControlFrame; | |
32 using spdy::SpdyGoAwayControlFrame; | |
33 using spdy::SpdyNoOpControlFrame; | |
34 using spdy::SpdyPingControlFrame; | |
35 using spdy::SpdyRstStreamControlFrame; | |
36 using spdy::SpdySettings; | |
37 using spdy::SpdySettingsControlFrame; | |
38 using spdy::SpdyStatusCodes; | |
39 using spdy::SpdySynReplyControlFrame; | |
40 using spdy::SpdySynStreamControlFrame; | |
41 using spdy::SpdyWindowUpdateControlFrame; | |
42 using spdy::SettingsFlagsAndId; | |
43 using spdy::kLengthMask; | |
44 using spdy::kSpdyProtocolVersion; | |
45 using spdy::kStreamIdMask; | |
46 | |
47 namespace { | |
48 | |
49 // Test our protocol constants | |
50 TEST(SpdyProtocolTest, ProtocolConstants) { | |
51 EXPECT_EQ(8u, SpdyFrame::kHeaderSize); | |
52 EXPECT_EQ(8u, SpdyDataFrame::size()); | |
53 EXPECT_EQ(8u, SpdyControlFrame::kHeaderSize); | |
54 EXPECT_EQ(18u, SpdySynStreamControlFrame::size()); | |
55 EXPECT_EQ(14u, SpdySynReplyControlFrame::size()); | |
56 EXPECT_EQ(16u, SpdyRstStreamControlFrame::size()); | |
57 EXPECT_EQ(12u, SpdySettingsControlFrame::size()); | |
58 EXPECT_EQ(8u, SpdyNoOpControlFrame::size()); | |
59 EXPECT_EQ(12u, SpdyPingControlFrame::size()); | |
60 EXPECT_EQ(12u, SpdyGoAwayControlFrame::size()); | |
61 EXPECT_EQ(14u, SpdyHeadersControlFrame::size()); | |
62 EXPECT_EQ(16u, SpdyWindowUpdateControlFrame::size()); | |
63 EXPECT_EQ(4u, sizeof(FlagsAndLength)); | |
64 EXPECT_EQ(1, SYN_STREAM); | |
65 EXPECT_EQ(2, SYN_REPLY); | |
66 EXPECT_EQ(3, RST_STREAM); | |
67 EXPECT_EQ(4, SETTINGS); | |
68 EXPECT_EQ(5, NOOP); | |
69 EXPECT_EQ(6, PING); | |
70 EXPECT_EQ(7, GOAWAY); | |
71 EXPECT_EQ(8, HEADERS); | |
72 EXPECT_EQ(9, WINDOW_UPDATE); | |
73 } | |
74 | |
75 // Test some of the protocol helper functions | |
76 TEST(SpdyProtocolTest, FrameStructs) { | |
77 SpdyFrame frame(SpdyFrame::kHeaderSize); | |
78 frame.set_length(12345); | |
79 frame.set_flags(10); | |
80 EXPECT_EQ(12345u, frame.length()); | |
81 EXPECT_EQ(10u, frame.flags()); | |
82 EXPECT_FALSE(frame.is_control_frame()); | |
83 | |
84 frame.set_length(0); | |
85 frame.set_flags(10); | |
86 EXPECT_EQ(0u, frame.length()); | |
87 EXPECT_EQ(10u, frame.flags()); | |
88 EXPECT_FALSE(frame.is_control_frame()); | |
89 } | |
90 | |
91 TEST(SpdyProtocolTest, DataFrameStructs) { | |
92 SpdyDataFrame data_frame; | |
93 data_frame.set_stream_id(12345); | |
94 EXPECT_EQ(12345u, data_frame.stream_id()); | |
95 } | |
96 | |
97 TEST(SpdyProtocolTest, ControlFrameStructs) { | |
98 SpdyFramer framer; | |
99 SpdyHeaderBlock headers; | |
100 | |
101 scoped_ptr<SpdySynStreamControlFrame> syn_frame( | |
102 framer.CreateSynStream(123, 456, 2, CONTROL_FLAG_FIN, false, &headers)); | |
103 EXPECT_EQ(kSpdyProtocolVersion, syn_frame->version()); | |
104 EXPECT_TRUE(syn_frame->is_control_frame()); | |
105 EXPECT_EQ(SYN_STREAM, syn_frame->type()); | |
106 EXPECT_EQ(123u, syn_frame->stream_id()); | |
107 EXPECT_EQ(456u, syn_frame->associated_stream_id()); | |
108 EXPECT_EQ(2u, syn_frame->priority()); | |
109 EXPECT_EQ(2, syn_frame->header_block_len()); | |
110 EXPECT_EQ(1u, syn_frame->flags()); | |
111 syn_frame->set_associated_stream_id(999u); | |
112 EXPECT_EQ(123u, syn_frame->stream_id()); | |
113 EXPECT_EQ(999u, syn_frame->associated_stream_id()); | |
114 | |
115 scoped_ptr<SpdySynReplyControlFrame> syn_reply( | |
116 framer.CreateSynReply(123, CONTROL_FLAG_NONE, false, &headers)); | |
117 EXPECT_EQ(kSpdyProtocolVersion, syn_reply->version()); | |
118 EXPECT_TRUE(syn_reply->is_control_frame()); | |
119 EXPECT_EQ(SYN_REPLY, syn_reply->type()); | |
120 EXPECT_EQ(123u, syn_reply->stream_id()); | |
121 EXPECT_EQ(2, syn_reply->header_block_len()); | |
122 EXPECT_EQ(0, syn_reply->flags()); | |
123 | |
124 scoped_ptr<SpdyRstStreamControlFrame> rst_frame( | |
125 framer.CreateRstStream(123, spdy::PROTOCOL_ERROR)); | |
126 EXPECT_EQ(kSpdyProtocolVersion, rst_frame->version()); | |
127 EXPECT_TRUE(rst_frame->is_control_frame()); | |
128 EXPECT_EQ(RST_STREAM, rst_frame->type()); | |
129 EXPECT_EQ(123u, rst_frame->stream_id()); | |
130 EXPECT_EQ(spdy::PROTOCOL_ERROR, rst_frame->status()); | |
131 rst_frame->set_status(spdy::INVALID_STREAM); | |
132 EXPECT_EQ(spdy::INVALID_STREAM, rst_frame->status()); | |
133 EXPECT_EQ(0, rst_frame->flags()); | |
134 | |
135 scoped_ptr<SpdyNoOpControlFrame> noop_frame( | |
136 framer.CreateNopFrame()); | |
137 EXPECT_EQ(kSpdyProtocolVersion, noop_frame->version()); | |
138 EXPECT_TRUE(noop_frame->is_control_frame()); | |
139 EXPECT_EQ(NOOP, noop_frame->type()); | |
140 EXPECT_EQ(0, noop_frame->flags()); | |
141 | |
142 const uint32 kUniqueId = 1234567u; | |
143 const uint32 kUniqueId2 = 31415926u; | |
144 scoped_ptr<SpdyPingControlFrame> ping_frame( | |
145 framer.CreatePingFrame(kUniqueId)); | |
146 EXPECT_EQ(kSpdyProtocolVersion, ping_frame->version()); | |
147 EXPECT_TRUE(ping_frame->is_control_frame()); | |
148 EXPECT_EQ(PING, ping_frame->type()); | |
149 EXPECT_EQ(kUniqueId, ping_frame->unique_id()); | |
150 ping_frame->set_unique_id(kUniqueId2); | |
151 EXPECT_EQ(kUniqueId2, ping_frame->unique_id()); | |
152 | |
153 scoped_ptr<SpdyGoAwayControlFrame> goaway_frame( | |
154 framer.CreateGoAway(123)); | |
155 EXPECT_EQ(kSpdyProtocolVersion, goaway_frame->version()); | |
156 EXPECT_TRUE(goaway_frame->is_control_frame()); | |
157 EXPECT_EQ(GOAWAY, goaway_frame->type()); | |
158 EXPECT_EQ(123u, goaway_frame->last_accepted_stream_id()); | |
159 | |
160 scoped_ptr<SpdyHeadersControlFrame> headers_frame( | |
161 framer.CreateHeaders(123, CONTROL_FLAG_NONE, false, &headers)); | |
162 EXPECT_EQ(kSpdyProtocolVersion, headers_frame->version()); | |
163 EXPECT_TRUE(headers_frame->is_control_frame()); | |
164 EXPECT_EQ(HEADERS, headers_frame->type()); | |
165 EXPECT_EQ(123u, headers_frame->stream_id()); | |
166 EXPECT_EQ(2, headers_frame->header_block_len()); | |
167 EXPECT_EQ(0, headers_frame->flags()); | |
168 | |
169 scoped_ptr<SpdyWindowUpdateControlFrame> window_update_frame( | |
170 framer.CreateWindowUpdate(123, 456)); | |
171 EXPECT_EQ(kSpdyProtocolVersion, window_update_frame->version()); | |
172 EXPECT_TRUE(window_update_frame->is_control_frame()); | |
173 EXPECT_EQ(WINDOW_UPDATE, window_update_frame->type()); | |
174 EXPECT_EQ(123u, window_update_frame->stream_id()); | |
175 EXPECT_EQ(456u, window_update_frame->delta_window_size()); | |
176 } | |
177 | |
178 TEST(SpdyProtocolTest, TestDataFrame) { | |
179 SpdyDataFrame frame; | |
180 | |
181 // Set the stream ID to various values. | |
182 frame.set_stream_id(0); | |
183 EXPECT_EQ(0u, frame.stream_id()); | |
184 EXPECT_FALSE(frame.is_control_frame()); | |
185 frame.set_stream_id(~0 & kStreamIdMask); | |
186 EXPECT_EQ(~0 & kStreamIdMask, frame.stream_id()); | |
187 EXPECT_FALSE(frame.is_control_frame()); | |
188 | |
189 // Set length to various values. Make sure that when you set_length(x), | |
190 // length() == x. Also make sure the flags are unaltered. | |
191 memset(frame.data(), '1', SpdyDataFrame::size()); | |
192 int8 flags = frame.flags(); | |
193 frame.set_length(0); | |
194 EXPECT_EQ(0u, frame.length()); | |
195 EXPECT_EQ(flags, frame.flags()); | |
196 frame.set_length(kLengthMask); | |
197 EXPECT_EQ(kLengthMask, frame.length()); | |
198 EXPECT_EQ(flags, frame.flags()); | |
199 frame.set_length(5u); | |
200 EXPECT_EQ(5u, frame.length()); | |
201 EXPECT_EQ(flags, frame.flags()); | |
202 | |
203 // Set flags to various values. Make sure that when you set_flags(x), | |
204 // flags() == x. Also make sure the length is unaltered. | |
205 memset(frame.data(), '1', SpdyDataFrame::size()); | |
206 uint32 length = frame.length(); | |
207 frame.set_flags(0u); | |
208 EXPECT_EQ(0u, frame.flags()); | |
209 EXPECT_EQ(length, frame.length()); | |
210 int8 all_flags = ~0; | |
211 frame.set_flags(all_flags); | |
212 flags = frame.flags(); | |
213 EXPECT_EQ(all_flags, flags); | |
214 EXPECT_EQ(length, frame.length()); | |
215 frame.set_flags(5u); | |
216 EXPECT_EQ(5u, frame.flags()); | |
217 EXPECT_EQ(length, frame.length()); | |
218 } | |
219 | |
220 // Test various types of SETTINGS frames. | |
221 TEST(SpdyProtocolTest, TestSpdySettingsFrame) { | |
222 SpdyFramer framer; | |
223 | |
224 // Create a settings frame with no settings. | |
225 SpdySettings settings; | |
226 scoped_ptr<SpdySettingsControlFrame> settings_frame( | |
227 framer.CreateSettings(settings)); | |
228 EXPECT_EQ(kSpdyProtocolVersion, settings_frame->version()); | |
229 EXPECT_TRUE(settings_frame->is_control_frame()); | |
230 EXPECT_EQ(SETTINGS, settings_frame->type()); | |
231 EXPECT_EQ(0u, settings_frame->num_entries()); | |
232 | |
233 // We'll add several different ID/Flag combinations and then verify | |
234 // that they encode and decode properly. | |
235 SettingsFlagsAndId ids[] = { | |
236 0x00000000, | |
237 0xffffffff, | |
238 0xff000001, | |
239 0x01000002, | |
240 }; | |
241 | |
242 for (size_t index = 0; index < arraysize(ids); ++index) { | |
243 settings.insert(settings.end(), std::make_pair(ids[index], index)); | |
244 settings_frame.reset(framer.CreateSettings(settings)); | |
245 EXPECT_EQ(kSpdyProtocolVersion, settings_frame->version()); | |
246 EXPECT_TRUE(settings_frame->is_control_frame()); | |
247 EXPECT_EQ(SETTINGS, settings_frame->type()); | |
248 EXPECT_EQ(index + 1, settings_frame->num_entries()); | |
249 | |
250 SpdySettings parsed_settings; | |
251 EXPECT_TRUE(framer.ParseSettings(settings_frame.get(), &parsed_settings)); | |
252 EXPECT_EQ(parsed_settings.size(), settings.size()); | |
253 SpdySettings::const_iterator it = parsed_settings.begin(); | |
254 int pos = 0; | |
255 while (it != parsed_settings.end()) { | |
256 SettingsFlagsAndId parsed = it->first; | |
257 uint32 value = it->second; | |
258 EXPECT_EQ(parsed.flags(), ids[pos].flags()); | |
259 EXPECT_EQ(parsed.id(), ids[pos].id()); | |
260 EXPECT_EQ(value, static_cast<uint32>(pos)); | |
261 ++it; | |
262 ++pos; | |
263 } | |
264 } | |
265 } | |
266 | |
267 TEST(SpdyProtocolTest, HasHeaderBlock) { | |
268 SpdyControlFrame frame(SpdyControlFrame::kHeaderSize); | |
269 for (SpdyControlType type = SYN_STREAM; | |
270 type < NUM_CONTROL_FRAME_TYPES; | |
271 type = static_cast<SpdyControlType>(type + 1)) { | |
272 frame.set_type(type); | |
273 if (type == SYN_STREAM || type == SYN_REPLY || type == HEADERS) { | |
274 EXPECT_TRUE(frame.has_header_block()); | |
275 } else { | |
276 EXPECT_FALSE(frame.has_header_block()); | |
277 } | |
278 } | |
279 } | |
280 | |
281 // Make sure that overflows both die in debug mode, and do not cause problems | |
282 // in opt mode. Note: The EXPECT_DEBUG_DEATH call does not work on Win32 yet, | |
283 // so we comment it out. | |
284 TEST(SpdyProtocolDeathTest, TestDataFrame) { | |
285 SpdyDataFrame frame; | |
286 | |
287 frame.set_stream_id(0); | |
288 // TODO(mbelshe): implement EXPECT_DEBUG_DEATH on windows. | |
289 #if !defined(WIN32) && defined(GTEST_HAS_DEATH_TEST) | |
290 #if !defined(DCHECK_ALWAYS_ON) | |
291 EXPECT_DEBUG_DEATH(frame.set_stream_id(~0), ""); | |
292 #else | |
293 EXPECT_DEATH(frame.set_stream_id(~0), ""); | |
294 #endif | |
295 #endif | |
296 EXPECT_FALSE(frame.is_control_frame()); | |
297 | |
298 frame.set_flags(0); | |
299 #if !defined(WIN32) && defined(GTEST_HAS_DEATH_TEST) | |
300 #if !defined(DCHECK_ALWAYS_ON) | |
301 EXPECT_DEBUG_DEATH(frame.set_length(~0), ""); | |
302 #else | |
303 EXPECT_DEATH(frame.set_length(~0), ""); | |
304 #endif | |
305 #endif | |
306 EXPECT_EQ(0, frame.flags()); | |
307 } | |
308 | |
309 TEST(SpdyProtocolDeathTest, TestSpdyControlFrameStreamId) { | |
310 SpdyControlFrame frame_store(SpdySynStreamControlFrame::size()); | |
311 memset(frame_store.data(), '1', SpdyControlFrame::kHeaderSize); | |
312 SpdySynStreamControlFrame* frame = | |
313 reinterpret_cast<SpdySynStreamControlFrame*>(&frame_store); | |
314 | |
315 // Set the stream ID to various values. | |
316 frame->set_stream_id(0); | |
317 EXPECT_EQ(0u, frame->stream_id()); | |
318 EXPECT_FALSE(frame->is_control_frame()); | |
319 frame->set_stream_id(kStreamIdMask); | |
320 EXPECT_EQ(kStreamIdMask, frame->stream_id()); | |
321 EXPECT_FALSE(frame->is_control_frame()); | |
322 } | |
323 | |
324 TEST(SpdyProtocolDeathTest, TestSpdyControlFrameVersion) { | |
325 const unsigned int kVersionMask = 0x7fff; | |
326 SpdyControlFrame frame(SpdySynStreamControlFrame::size()); | |
327 memset(frame.data(), '1', SpdyControlFrame::kHeaderSize); | |
328 | |
329 // Set the version to various values, and make sure it does not affect the | |
330 // type. | |
331 frame.set_type(SYN_STREAM); | |
332 frame.set_version(0); | |
333 EXPECT_EQ(0, frame.version()); | |
334 EXPECT_TRUE(frame.is_control_frame()); | |
335 EXPECT_EQ(SYN_STREAM, frame.type()); | |
336 | |
337 SpdySynStreamControlFrame* syn_stream = | |
338 reinterpret_cast<SpdySynStreamControlFrame*>(&frame); | |
339 syn_stream->set_stream_id(~0 & kVersionMask); | |
340 EXPECT_EQ(~0 & kVersionMask, syn_stream->stream_id()); | |
341 EXPECT_TRUE(frame.is_control_frame()); | |
342 EXPECT_EQ(SYN_STREAM, frame.type()); | |
343 } | |
344 | |
345 TEST(SpdyProtocolDeathTest, TestSpdyControlFrameType) { | |
346 SpdyControlFrame frame(SpdyControlFrame::kHeaderSize); | |
347 memset(frame.data(), 255, SpdyControlFrame::kHeaderSize); | |
348 | |
349 // type() should be out of bounds. | |
350 EXPECT_FALSE(frame.AppearsToBeAValidControlFrame()); | |
351 | |
352 uint16 version = frame.version(); | |
353 | |
354 for (int i = SYN_STREAM; i <= spdy::NOOP; ++i) { | |
355 frame.set_type(static_cast<SpdyControlType>(i)); | |
356 EXPECT_EQ(i, static_cast<int>(frame.type())); | |
357 EXPECT_TRUE(frame.AppearsToBeAValidControlFrame()); | |
358 // Make sure setting type does not alter the version block. | |
359 EXPECT_EQ(version, frame.version()); | |
360 EXPECT_TRUE(frame.is_control_frame()); | |
361 } | |
362 } | |
363 | |
364 TEST(SpdyProtocolDeathTest, TestRstStreamStatusBounds) { | |
365 SpdyFramer framer; | |
366 scoped_ptr<SpdyRstStreamControlFrame> rst_frame; | |
367 | |
368 rst_frame.reset(framer.CreateRstStream(123, spdy::PROTOCOL_ERROR)); | |
369 EXPECT_EQ(spdy::PROTOCOL_ERROR, rst_frame->status()); | |
370 | |
371 rst_frame->set_status(spdy::INVALID); | |
372 EXPECT_EQ(spdy::INVALID, rst_frame->status()); | |
373 | |
374 rst_frame->set_status( | |
375 static_cast<spdy::SpdyStatusCodes>(spdy::INVALID - 1)); | |
376 EXPECT_EQ(spdy::INVALID, rst_frame->status()); | |
377 | |
378 rst_frame->set_status(spdy::NUM_STATUS_CODES); | |
379 EXPECT_EQ(spdy::INVALID, rst_frame->status()); | |
380 } | |
381 | |
382 } // namespace | |
OLD | NEW |