| 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 OnSynStream(SpdyStreamId stream_id, |
| 27 SpdyStreamId associated_stream_id, |
| 28 SpdyPriority priority, |
| 29 uint8 credential_slot, |
| 30 bool fin, |
| 31 bool unidirectional) OVERRIDE {} |
| 32 virtual void OnSynReply(SpdyStreamId stream_id, bool fin) OVERRIDE {} |
| 33 virtual void OnHeaders(SpdyStreamId stream_id, bool fin) OVERRIDE {} |
| 34 virtual bool OnControlFrameHeaderData(SpdyStreamId stream_id, |
| 35 const char* header_data, |
| 36 size_t len) OVERRIDE; |
| 37 virtual bool OnCredentialFrameData(const char* credential_data, |
| 38 size_t len) OVERRIDE { |
| 39 return false; |
| 40 } |
| 41 virtual void OnDataFrameHeader(SpdyStreamId stream_id, |
| 42 size_t length, |
| 43 bool fin) OVERRIDE {} |
| 44 virtual void OnStreamFrameData(SpdyStreamId stream_id, |
| 45 const char* data, |
| 46 size_t len, |
| 47 bool fin) OVERRIDE {} |
| 48 virtual void OnSetting(SpdySettingsIds id, |
| 49 uint8 flags, |
| 50 uint32 value) OVERRIDE {} |
| 51 virtual void OnPing(uint32 unique_id) OVERRIDE {} |
| 52 virtual void OnRstStream(SpdyStreamId stream_id, |
| 53 SpdyRstStreamStatus status) OVERRIDE {} |
| 54 virtual void OnGoAway(SpdyStreamId last_accepted_stream_id, |
| 55 SpdyGoAwayStatus status) OVERRIDE {} |
| 56 virtual void OnWindowUpdate(SpdyStreamId stream_id, |
| 57 uint32 delta_window_size) OVERRIDE {} |
| 58 virtual void OnSynStreamCompressed(size_t uncompressed_size, |
| 59 size_t compressed_size) OVERRIDE {} |
| 60 void set_visitor(QuicSpdyDecompressor::Visitor* visitor) { |
| 61 DCHECK(visitor); |
| 62 visitor_ = visitor; |
| 63 } |
| 64 |
| 65 private: |
| 66 QuicSpdyDecompressor::Visitor* visitor_; |
| 67 bool error_; |
| 68 }; |
| 69 |
| 70 bool SpdyFramerVisitor::OnControlFrameHeaderData(SpdyStreamId stream_id, |
| 71 const char* header_data, |
| 72 size_t len) { |
| 73 DCHECK(visitor_); |
| 74 return visitor_->OnDecompressedData(StringPiece(header_data, len)); |
| 75 } |
| 76 |
| 77 QuicSpdyDecompressor::QuicSpdyDecompressor() |
| 78 : spdy_framer_(3), |
| 79 spdy_visitor_(new SpdyFramerVisitor(NULL)), |
| 80 current_header_id_(1), |
| 81 has_current_compressed_size_(false), |
| 82 current_compressed_size_(0), |
| 83 compressed_bytes_consumed_(0) { |
| 84 spdy_framer_.set_visitor(spdy_visitor_.get()); |
| 85 } |
| 86 |
| 87 QuicSpdyDecompressor::~QuicSpdyDecompressor() { |
| 88 } |
| 89 |
| 90 size_t QuicSpdyDecompressor::DecompressData(StringPiece data, |
| 91 Visitor* visitor) { |
| 92 spdy_visitor_->set_visitor(visitor); |
| 93 size_t bytes_consumed = 0; |
| 94 |
| 95 if (!has_current_compressed_size_) { |
| 96 const size_t kCompressedBufferSizeSize = sizeof(uint32); |
| 97 DCHECK_GT(kCompressedBufferSizeSize, compressed_size_buffer_.length()); |
| 98 size_t missing_size = |
| 99 kCompressedBufferSizeSize - compressed_size_buffer_.length(); |
| 100 if (data.length() < missing_size) { |
| 101 data.AppendToString(&compressed_size_buffer_); |
| 102 return data.length(); |
| 103 } |
| 104 bytes_consumed += missing_size; |
| 105 data.substr(0, missing_size).AppendToString(&compressed_size_buffer_); |
| 106 DCHECK_EQ(kCompressedBufferSizeSize, compressed_size_buffer_.length()); |
| 107 memcpy(¤t_compressed_size_, compressed_size_buffer_.data(), |
| 108 kCompressedBufferSizeSize); |
| 109 compressed_size_buffer_.clear(); |
| 110 has_current_compressed_size_ = true; |
| 111 data = data.substr(missing_size); |
| 112 compressed_bytes_consumed_ = 0; |
| 113 } |
| 114 |
| 115 size_t bytes_to_consume = |
| 116 min(current_compressed_size_ - compressed_bytes_consumed_, |
| 117 static_cast<uint32>(data.length())); |
| 118 if (bytes_to_consume > 0) { |
| 119 bool success = spdy_framer_.IncrementallyDecompressControlFrameHeaderData( |
| 120 current_header_id_, data.data(), bytes_to_consume); |
| 121 DCHECK(success); |
| 122 compressed_bytes_consumed_ += bytes_to_consume; |
| 123 bytes_consumed += bytes_to_consume; |
| 124 } |
| 125 if (current_compressed_size_ - compressed_bytes_consumed_ == 0) { |
| 126 ResetForNextHeaders(); |
| 127 } |
| 128 return bytes_consumed; |
| 129 } |
| 130 |
| 131 void QuicSpdyDecompressor::ResetForNextHeaders() { |
| 132 has_current_compressed_size_ = false; |
| 133 current_compressed_size_ = 0; |
| 134 compressed_bytes_consumed_ = 0; |
| 135 ++current_header_id_; |
| 136 } |
| 137 |
| 138 } // namespace net |
| OLD | NEW |