| OLD | NEW |
| (Empty) |
| 1 // Copyright 2016 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/http2/decoder/decode_http2_structures.h" | |
| 6 | |
| 7 #include <string.h> | |
| 8 | |
| 9 #include "base/logging.h" | |
| 10 #include "net/http2/decoder/decode_buffer.h" | |
| 11 #include "net/http2/http2_constants.h" | |
| 12 | |
| 13 namespace net { | |
| 14 | |
| 15 // Http2FrameHeader decoding: | |
| 16 | |
| 17 void DoDecode(Http2FrameHeader* out, DecodeBuffer* b) { | |
| 18 DCHECK_NE(nullptr, out); | |
| 19 DCHECK_NE(nullptr, b); | |
| 20 DCHECK_LE(Http2FrameHeader::EncodedSize(), b->Remaining()); | |
| 21 out->payload_length = b->DecodeUInt24(); | |
| 22 out->type = static_cast<Http2FrameType>(b->DecodeUInt8()); | |
| 23 out->flags = static_cast<Http2FrameFlag>(b->DecodeUInt8()); | |
| 24 out->stream_id = b->DecodeUInt31(); | |
| 25 } | |
| 26 | |
| 27 bool MaybeDecode(Http2FrameHeader* out, DecodeBuffer* b) { | |
| 28 DCHECK_NE(nullptr, out); | |
| 29 DCHECK_NE(nullptr, b); | |
| 30 if (b->Remaining() >= Http2FrameHeader::EncodedSize()) { | |
| 31 DoDecode(out, b); | |
| 32 return true; | |
| 33 } | |
| 34 return false; | |
| 35 } | |
| 36 | |
| 37 bool SlowDecode(Http2FrameHeader* out, DecodeBuffer* b, uint32_t* offset) { | |
| 38 DCHECK_NE(nullptr, out); | |
| 39 DCHECK_NE(nullptr, b); | |
| 40 DCHECK_NE(nullptr, offset); | |
| 41 DCHECK_GT(Http2FrameHeader::EncodedSize(), *offset); | |
| 42 if (b->SlowDecodeUInt24(0 /* field_offset */, offset, &out->payload_length) && | |
| 43 b->SlowDecodeEnum(1 /* field_size */, 3 /* field_offset */, offset, | |
| 44 &out->type) && | |
| 45 b->SlowDecodeEnum(1 /* field_size */, 4 /* field_offset */, offset, | |
| 46 &out->flags) && | |
| 47 b->SlowDecodeUInt31(5 /* field_offset */, offset, &out->stream_id)) { | |
| 48 DCHECK_EQ(Http2FrameHeader::EncodedSize(), *offset); | |
| 49 return true; | |
| 50 } | |
| 51 DCHECK_GT(Http2FrameHeader::EncodedSize(), *offset); | |
| 52 return false; | |
| 53 } | |
| 54 | |
| 55 // Http2PriorityFields decoding: | |
| 56 | |
| 57 void DoDecode(Http2PriorityFields* out, DecodeBuffer* b) { | |
| 58 DCHECK_NE(nullptr, out); | |
| 59 DCHECK_NE(nullptr, b); | |
| 60 DCHECK_LE(Http2PriorityFields::EncodedSize(), b->Remaining()); | |
| 61 uint32_t stream_id_and_flag = b->DecodeUInt32(); | |
| 62 out->stream_dependency = stream_id_and_flag & StreamIdMask(); | |
| 63 if (out->stream_dependency == stream_id_and_flag) { | |
| 64 out->is_exclusive = false; | |
| 65 } else { | |
| 66 out->is_exclusive = true; | |
| 67 } | |
| 68 // Note that chars are automatically promoted to ints during arithmetic, | |
| 69 // so 255 + 1 doesn't end up as zero. | |
| 70 out->weight = b->DecodeUInt8() + 1; | |
| 71 } | |
| 72 | |
| 73 bool MaybeDecode(Http2PriorityFields* out, DecodeBuffer* b) { | |
| 74 DCHECK_NE(nullptr, out); | |
| 75 DCHECK_NE(nullptr, b); | |
| 76 if (b->Remaining() >= Http2PriorityFields::EncodedSize()) { | |
| 77 DoDecode(out, b); | |
| 78 return true; | |
| 79 } | |
| 80 return false; | |
| 81 } | |
| 82 | |
| 83 bool SlowDecode(Http2PriorityFields* out, DecodeBuffer* b, uint32_t* offset) { | |
| 84 DCHECK_NE(nullptr, out); | |
| 85 DCHECK_NE(nullptr, b); | |
| 86 DCHECK_NE(nullptr, offset); | |
| 87 DCHECK_GT(Http2PriorityFields::EncodedSize(), *offset); | |
| 88 const uint32_t start_offset = *offset; | |
| 89 if (b->SlowDecodeUInt32(0 /* field_offset */, offset, | |
| 90 &out->stream_dependency) && | |
| 91 b->SlowDecodeUnsignedInt(1, // field_size | |
| 92 4, // field_offset | |
| 93 offset, &out->weight)) { | |
| 94 DCHECK_EQ(Http2PriorityFields::EncodedSize(), *offset); | |
| 95 if (start_offset < *offset) { | |
| 96 // First time here. Extract is_exclusive from stream_dependency. | |
| 97 const uint32_t stream_id_only = out->stream_dependency & StreamIdMask(); | |
| 98 if (out->stream_dependency != stream_id_only) { | |
| 99 out->stream_dependency = stream_id_only; | |
| 100 out->is_exclusive = true; | |
| 101 } else { | |
| 102 out->is_exclusive = false; | |
| 103 } | |
| 104 // Need to add one to the weight field because the encoding is 0-255, but | |
| 105 // interpreted as 1-256. | |
| 106 ++(out->weight); | |
| 107 } | |
| 108 return true; | |
| 109 } | |
| 110 DCHECK_GT(Http2PriorityFields::EncodedSize(), *offset); | |
| 111 return false; | |
| 112 } | |
| 113 | |
| 114 // Http2RstStreamFields decoding: | |
| 115 | |
| 116 void DoDecode(Http2RstStreamFields* out, DecodeBuffer* b) { | |
| 117 DCHECK_NE(nullptr, out); | |
| 118 DCHECK_NE(nullptr, b); | |
| 119 DCHECK_LE(Http2RstStreamFields::EncodedSize(), b->Remaining()); | |
| 120 out->error_code = static_cast<Http2ErrorCode>(b->DecodeUInt32()); | |
| 121 } | |
| 122 | |
| 123 bool MaybeDecode(Http2RstStreamFields* out, DecodeBuffer* b) { | |
| 124 DCHECK_NE(nullptr, out); | |
| 125 DCHECK_NE(nullptr, b); | |
| 126 if (b->Remaining() >= Http2RstStreamFields::EncodedSize()) { | |
| 127 DoDecode(out, b); | |
| 128 return true; | |
| 129 } | |
| 130 return false; | |
| 131 } | |
| 132 | |
| 133 bool SlowDecode(Http2RstStreamFields* out, DecodeBuffer* b, uint32_t* offset) { | |
| 134 DCHECK_NE(nullptr, out); | |
| 135 DCHECK_NE(nullptr, b); | |
| 136 DCHECK_NE(nullptr, offset); | |
| 137 DCHECK_GT(Http2RstStreamFields::EncodedSize(), *offset); | |
| 138 | |
| 139 if (b->SlowDecodeEnum(4 /* field_size */, 0 /* field_offset */, offset, | |
| 140 &out->error_code)) { | |
| 141 DCHECK_EQ(Http2RstStreamFields::EncodedSize(), *offset); | |
| 142 return true; | |
| 143 } | |
| 144 DCHECK_GT(Http2RstStreamFields::EncodedSize(), *offset); | |
| 145 return false; | |
| 146 } | |
| 147 | |
| 148 // Http2SettingFields decoding: | |
| 149 | |
| 150 void DoDecode(Http2SettingFields* out, DecodeBuffer* b) { | |
| 151 DCHECK_NE(nullptr, out); | |
| 152 DCHECK_NE(nullptr, b); | |
| 153 DCHECK_LE(Http2SettingFields::EncodedSize(), b->Remaining()); | |
| 154 out->parameter = static_cast<Http2SettingsParameter>(b->DecodeUInt16()); | |
| 155 out->value = b->DecodeUInt32(); | |
| 156 } | |
| 157 | |
| 158 bool MaybeDecode(Http2SettingFields* out, DecodeBuffer* b) { | |
| 159 DCHECK_NE(nullptr, out); | |
| 160 DCHECK_NE(nullptr, b); | |
| 161 if (b->Remaining() >= Http2SettingFields::EncodedSize()) { | |
| 162 DoDecode(out, b); | |
| 163 return true; | |
| 164 } | |
| 165 return false; | |
| 166 } | |
| 167 | |
| 168 bool SlowDecode(Http2SettingFields* out, DecodeBuffer* b, uint32_t* offset) { | |
| 169 DCHECK_NE(nullptr, out); | |
| 170 DCHECK_NE(nullptr, b); | |
| 171 DCHECK_NE(nullptr, offset); | |
| 172 DCHECK_LT(*offset, Http2SettingFields::EncodedSize()); | |
| 173 | |
| 174 if (b->SlowDecodeEnum(2 /* field_size */, 0 /* field_offset */, offset, | |
| 175 &out->parameter) && | |
| 176 b->SlowDecodeUInt32(2 /* field_offset */, offset, &out->value)) { | |
| 177 DCHECK_EQ(Http2SettingFields::EncodedSize(), *offset); | |
| 178 return true; | |
| 179 } | |
| 180 DCHECK_LT(*offset, Http2SettingFields::EncodedSize()); | |
| 181 return false; | |
| 182 } | |
| 183 | |
| 184 // Http2PushPromiseFields decoding: | |
| 185 | |
| 186 void DoDecode(Http2PushPromiseFields* out, DecodeBuffer* b) { | |
| 187 DCHECK_NE(nullptr, out); | |
| 188 DCHECK_NE(nullptr, b); | |
| 189 DCHECK_LE(Http2PushPromiseFields::EncodedSize(), b->Remaining()); | |
| 190 out->promised_stream_id = b->DecodeUInt31(); | |
| 191 } | |
| 192 | |
| 193 bool MaybeDecode(Http2PushPromiseFields* out, DecodeBuffer* b) { | |
| 194 DCHECK_NE(nullptr, out); | |
| 195 DCHECK_NE(nullptr, b); | |
| 196 if (b->Remaining() >= Http2PushPromiseFields::EncodedSize()) { | |
| 197 DoDecode(out, b); | |
| 198 return true; | |
| 199 } | |
| 200 return false; | |
| 201 } | |
| 202 | |
| 203 bool SlowDecode(Http2PushPromiseFields* out, | |
| 204 DecodeBuffer* b, | |
| 205 uint32_t* offset) { | |
| 206 DCHECK_NE(nullptr, out); | |
| 207 DCHECK_NE(nullptr, b); | |
| 208 DCHECK_NE(nullptr, offset); | |
| 209 DCHECK_LT(*offset, Http2PushPromiseFields::EncodedSize()); | |
| 210 if (b->SlowDecodeUInt31(0 /* field_offset */, offset, | |
| 211 &out->promised_stream_id)) { | |
| 212 DCHECK_EQ(Http2PushPromiseFields::EncodedSize(), *offset); | |
| 213 return true; | |
| 214 } | |
| 215 DCHECK_LT(*offset, Http2PushPromiseFields::EncodedSize()); | |
| 216 return false; | |
| 217 } | |
| 218 | |
| 219 // Http2PingFields decoding: | |
| 220 | |
| 221 void DoDecode(Http2PingFields* out, DecodeBuffer* b) { | |
| 222 DCHECK_NE(nullptr, out); | |
| 223 DCHECK_NE(nullptr, b); | |
| 224 DCHECK_LE(Http2PingFields::EncodedSize(), b->Remaining()); | |
| 225 memcpy(out->opaque_data, b->cursor(), Http2PingFields::EncodedSize()); | |
| 226 b->AdvanceCursor(Http2PingFields::EncodedSize()); | |
| 227 } | |
| 228 | |
| 229 bool MaybeDecode(Http2PingFields* out, DecodeBuffer* b) { | |
| 230 DCHECK_NE(nullptr, out); | |
| 231 DCHECK_NE(nullptr, b); | |
| 232 if (b->Remaining() >= Http2PingFields::EncodedSize()) { | |
| 233 DoDecode(out, b); | |
| 234 return true; | |
| 235 } | |
| 236 return false; | |
| 237 } | |
| 238 | |
| 239 bool SlowDecode(Http2PingFields* out, DecodeBuffer* b, uint32_t* offset) { | |
| 240 DCHECK_NE(nullptr, out); | |
| 241 DCHECK_NE(nullptr, b); | |
| 242 DCHECK_NE(nullptr, offset); | |
| 243 DCHECK_LT(*offset, Http2PingFields::EncodedSize()); | |
| 244 while (*offset < Http2PingFields::EncodedSize()) { | |
| 245 if (b->Empty()) { | |
| 246 return false; | |
| 247 } | |
| 248 out->opaque_data[(*offset)++] = b->DecodeUInt8(); | |
| 249 } | |
| 250 return true; | |
| 251 } | |
| 252 | |
| 253 // Http2GoAwayFields decoding: | |
| 254 | |
| 255 void DoDecode(Http2GoAwayFields* out, DecodeBuffer* b) { | |
| 256 DCHECK_NE(nullptr, out); | |
| 257 DCHECK_NE(nullptr, b); | |
| 258 DCHECK_LE(Http2GoAwayFields::EncodedSize(), b->Remaining()); | |
| 259 out->last_stream_id = b->DecodeUInt31(); | |
| 260 out->error_code = static_cast<Http2ErrorCode>(b->DecodeUInt32()); | |
| 261 } | |
| 262 | |
| 263 bool MaybeDecode(Http2GoAwayFields* out, DecodeBuffer* b) { | |
| 264 DCHECK_NE(nullptr, out); | |
| 265 DCHECK_NE(nullptr, b); | |
| 266 if (b->Remaining() >= Http2GoAwayFields::EncodedSize()) { | |
| 267 DoDecode(out, b); | |
| 268 return true; | |
| 269 } | |
| 270 return false; | |
| 271 } | |
| 272 | |
| 273 bool SlowDecode(Http2GoAwayFields* out, DecodeBuffer* b, uint32_t* offset) { | |
| 274 DCHECK_NE(nullptr, out); | |
| 275 DCHECK_NE(nullptr, b); | |
| 276 DCHECK_NE(nullptr, offset); | |
| 277 DCHECK_LT(*offset, Http2GoAwayFields::EncodedSize()); | |
| 278 if (b->SlowDecodeUInt31(0 /* field_offset */, offset, &out->last_stream_id) && | |
| 279 b->SlowDecodeEnum(4 /* field_size */, 4 /* field_offset */, offset, | |
| 280 &out->error_code)) { | |
| 281 DCHECK_EQ(Http2GoAwayFields::EncodedSize(), *offset); | |
| 282 return true; | |
| 283 } | |
| 284 DCHECK_LT(*offset, Http2GoAwayFields::EncodedSize()); | |
| 285 return false; | |
| 286 } | |
| 287 | |
| 288 // Http2WindowUpdateFields decoding: | |
| 289 | |
| 290 void DoDecode(Http2WindowUpdateFields* out, DecodeBuffer* b) { | |
| 291 DCHECK_NE(nullptr, out); | |
| 292 DCHECK_NE(nullptr, b); | |
| 293 DCHECK_LE(Http2WindowUpdateFields::EncodedSize(), b->Remaining()); | |
| 294 out->window_size_increment = b->DecodeUInt31(); | |
| 295 } | |
| 296 | |
| 297 bool MaybeDecode(Http2WindowUpdateFields* out, DecodeBuffer* b) { | |
| 298 DCHECK_NE(nullptr, out); | |
| 299 DCHECK_NE(nullptr, b); | |
| 300 if (b->Remaining() >= Http2WindowUpdateFields::EncodedSize()) { | |
| 301 DoDecode(out, b); | |
| 302 return true; | |
| 303 } | |
| 304 return false; | |
| 305 } | |
| 306 | |
| 307 bool SlowDecode(Http2WindowUpdateFields* out, | |
| 308 DecodeBuffer* b, | |
| 309 uint32_t* offset) { | |
| 310 DCHECK_NE(nullptr, out); | |
| 311 DCHECK_NE(nullptr, b); | |
| 312 DCHECK_NE(nullptr, offset); | |
| 313 DCHECK_LT(*offset, Http2WindowUpdateFields::EncodedSize()); | |
| 314 if (b->SlowDecodeUInt31(0 /* field_offset */, offset, | |
| 315 &out->window_size_increment)) { | |
| 316 DCHECK_EQ(Http2WindowUpdateFields::EncodedSize(), *offset); | |
| 317 return true; | |
| 318 } | |
| 319 DCHECK_LT(*offset, Http2WindowUpdateFields::EncodedSize()); | |
| 320 return false; | |
| 321 } | |
| 322 | |
| 323 // Http2AltSvcFields decoding: | |
| 324 | |
| 325 void DoDecode(Http2AltSvcFields* out, DecodeBuffer* b) { | |
| 326 DCHECK_NE(nullptr, out); | |
| 327 DCHECK_NE(nullptr, b); | |
| 328 DCHECK_LE(Http2AltSvcFields::EncodedSize(), b->Remaining()); | |
| 329 out->origin_length = b->DecodeUInt16(); | |
| 330 } | |
| 331 | |
| 332 bool MaybeDecode(Http2AltSvcFields* out, DecodeBuffer* b) { | |
| 333 DCHECK_NE(nullptr, out); | |
| 334 DCHECK_NE(nullptr, b); | |
| 335 if (b->Remaining() >= Http2AltSvcFields::EncodedSize()) { | |
| 336 DoDecode(out, b); | |
| 337 return true; | |
| 338 } | |
| 339 return false; | |
| 340 } | |
| 341 | |
| 342 bool SlowDecode(Http2AltSvcFields* out, DecodeBuffer* b, uint32_t* offset) { | |
| 343 DCHECK_NE(nullptr, out); | |
| 344 DCHECK_NE(nullptr, b); | |
| 345 DCHECK_NE(nullptr, offset); | |
| 346 DCHECK_LT(*offset, Http2AltSvcFields::EncodedSize()); | |
| 347 if (b->SlowDecodeUInt16(0 /* field_offset */, offset, &out->origin_length)) { | |
| 348 DCHECK_EQ(Http2AltSvcFields::EncodedSize(), *offset); | |
| 349 return true; | |
| 350 } | |
| 351 DCHECK_LT(*offset, Http2AltSvcFields::EncodedSize()); | |
| 352 return false; | |
| 353 } | |
| 354 | |
| 355 } // namespace net | |
| OLD | NEW |