| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2013 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/quic/quic_spdy_decompressor.h" | |
| 6 | |
| 7 #include <algorithm> | |
| 8 | |
| 9 #include "base/logging.h" | |
| 10 | |
| 11 using base::StringPiece; | |
| 12 using std::min; | |
| 13 | |
| 14 namespace net { | |
| 15 | |
| 16 class SpdyFramerVisitor : public SpdyFramerVisitorInterface { | |
| 17 public: | |
| 18 explicit SpdyFramerVisitor(QuicSpdyDecompressor::Visitor* visitor) | |
| 19 : visitor_(visitor), | |
| 20 error_(false) { | |
| 21 } | |
| 22 | |
| 23 virtual void OnError(SpdyFramer* framer) OVERRIDE { | |
| 24 error_ = true; | |
| 25 } | |
| 26 virtual void OnDataFrameHeader(SpdyStreamId stream_id, | |
| 27 size_t length, | |
| 28 bool fin) OVERRIDE {} | |
| 29 virtual void OnStreamFrameData(SpdyStreamId stream_id, | |
| 30 const char* data, | |
| 31 size_t len, | |
| 32 bool fin) OVERRIDE {} | |
| 33 virtual bool OnControlFrameHeaderData(SpdyStreamId stream_id, | |
| 34 const char* header_data, | |
| 35 size_t len) OVERRIDE; | |
| 36 virtual void OnSynStream(SpdyStreamId stream_id, | |
| 37 SpdyStreamId associated_stream_id, | |
| 38 SpdyPriority priority, | |
| 39 bool fin, | |
| 40 bool unidirectional) OVERRIDE {} | |
| 41 virtual void OnSynReply(SpdyStreamId stream_id, bool fin) OVERRIDE {} | |
| 42 virtual void OnRstStream(SpdyStreamId stream_id, | |
| 43 SpdyRstStreamStatus status) OVERRIDE {} | |
| 44 virtual void OnSetting(SpdySettingsIds id, | |
| 45 uint8 flags, | |
| 46 uint32 value) OVERRIDE {} | |
| 47 virtual void OnSettingsAck() OVERRIDE {} | |
| 48 virtual void OnSettingsEnd() OVERRIDE {} | |
| 49 virtual void OnPing(SpdyPingId unique_id, bool is_ack) OVERRIDE {} | |
| 50 virtual void OnGoAway(SpdyStreamId last_accepted_stream_id, | |
| 51 SpdyGoAwayStatus status) OVERRIDE {} | |
| 52 virtual void OnHeaders(SpdyStreamId stream_id, bool fin) OVERRIDE {} | |
| 53 virtual void OnWindowUpdate(SpdyStreamId stream_id, | |
| 54 uint32 delta_window_size) OVERRIDE {} | |
| 55 virtual void OnPushPromise(SpdyStreamId stream_id, | |
| 56 SpdyStreamId promised_stream_id) OVERRIDE {} | |
| 57 void set_visitor(QuicSpdyDecompressor::Visitor* visitor) { | |
| 58 DCHECK(visitor); | |
| 59 visitor_ = visitor; | |
| 60 } | |
| 61 | |
| 62 private: | |
| 63 QuicSpdyDecompressor::Visitor* visitor_; | |
| 64 bool error_; | |
| 65 }; | |
| 66 | |
| 67 bool SpdyFramerVisitor::OnControlFrameHeaderData(SpdyStreamId stream_id, | |
| 68 const char* header_data, | |
| 69 size_t len) { | |
| 70 DCHECK(visitor_); | |
| 71 return visitor_->OnDecompressedData(StringPiece(header_data, len)); | |
| 72 } | |
| 73 | |
| 74 QuicSpdyDecompressor::QuicSpdyDecompressor() | |
| 75 : spdy_framer_(SPDY3), | |
| 76 spdy_visitor_(new SpdyFramerVisitor(NULL)), | |
| 77 current_header_id_(1), | |
| 78 has_current_compressed_size_(false), | |
| 79 current_compressed_size_(0), | |
| 80 compressed_bytes_consumed_(0) { | |
| 81 spdy_framer_.set_visitor(spdy_visitor_.get()); | |
| 82 } | |
| 83 | |
| 84 QuicSpdyDecompressor::~QuicSpdyDecompressor() { | |
| 85 } | |
| 86 | |
| 87 size_t QuicSpdyDecompressor::DecompressData(StringPiece data, | |
| 88 Visitor* visitor) { | |
| 89 spdy_visitor_->set_visitor(visitor); | |
| 90 size_t bytes_consumed = 0; | |
| 91 | |
| 92 if (!has_current_compressed_size_) { | |
| 93 const size_t kCompressedBufferSizeSize = sizeof(uint32); | |
| 94 DCHECK_GT(kCompressedBufferSizeSize, compressed_size_buffer_.length()); | |
| 95 size_t missing_size = | |
| 96 kCompressedBufferSizeSize - compressed_size_buffer_.length(); | |
| 97 if (data.length() < missing_size) { | |
| 98 data.AppendToString(&compressed_size_buffer_); | |
| 99 return data.length(); | |
| 100 } | |
| 101 bytes_consumed += missing_size; | |
| 102 data.substr(0, missing_size).AppendToString(&compressed_size_buffer_); | |
| 103 DCHECK_EQ(kCompressedBufferSizeSize, compressed_size_buffer_.length()); | |
| 104 memcpy(¤t_compressed_size_, compressed_size_buffer_.data(), | |
| 105 kCompressedBufferSizeSize); | |
| 106 compressed_size_buffer_.clear(); | |
| 107 has_current_compressed_size_ = true; | |
| 108 data = data.substr(missing_size); | |
| 109 compressed_bytes_consumed_ = 0; | |
| 110 } | |
| 111 | |
| 112 size_t bytes_to_consume = | |
| 113 min(current_compressed_size_ - compressed_bytes_consumed_, | |
| 114 static_cast<uint32>(data.length())); | |
| 115 if (bytes_to_consume > 0) { | |
| 116 if (!spdy_framer_.IncrementallyDecompressControlFrameHeaderData( | |
| 117 current_header_id_, data.data(), bytes_to_consume)) { | |
| 118 visitor->OnDecompressionError(); | |
| 119 return bytes_consumed; | |
| 120 } | |
| 121 compressed_bytes_consumed_ += bytes_to_consume; | |
| 122 bytes_consumed += bytes_to_consume; | |
| 123 } | |
| 124 if (current_compressed_size_ - compressed_bytes_consumed_ == 0) { | |
| 125 ResetForNextHeaders(); | |
| 126 } | |
| 127 return bytes_consumed; | |
| 128 } | |
| 129 | |
| 130 void QuicSpdyDecompressor::ResetForNextHeaders() { | |
| 131 has_current_compressed_size_ = false; | |
| 132 current_compressed_size_ = 0; | |
| 133 compressed_bytes_consumed_ = 0; | |
| 134 ++current_header_id_; | |
| 135 } | |
| 136 | |
| 137 } // namespace net | |
| OLD | NEW |