xref: /aosp_15_r20/external/cronet/net/websockets/websocket_deflate_stream.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2013 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_deflate_stream.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <stdint.h>
8*6777b538SAndroid Build Coastguard Worker 
9*6777b538SAndroid Build Coastguard Worker #include <algorithm>
10*6777b538SAndroid Build Coastguard Worker #include <ostream>
11*6777b538SAndroid Build Coastguard Worker #include <string>
12*6777b538SAndroid Build Coastguard Worker #include <utility>
13*6777b538SAndroid Build Coastguard Worker #include <vector>
14*6777b538SAndroid Build Coastguard Worker 
15*6777b538SAndroid Build Coastguard Worker #include "base/check.h"
16*6777b538SAndroid Build Coastguard Worker #include "base/check_op.h"
17*6777b538SAndroid Build Coastguard Worker #include "base/functional/bind.h"
18*6777b538SAndroid Build Coastguard Worker #include "base/functional/callback.h"
19*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
20*6777b538SAndroid Build Coastguard Worker #include "base/memory/scoped_refptr.h"
21*6777b538SAndroid Build Coastguard Worker #include "base/notreached.h"
22*6777b538SAndroid Build Coastguard Worker #include "net/base/io_buffer.h"
23*6777b538SAndroid Build Coastguard Worker #include "net/base/net_errors.h"
24*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_deflate_parameters.h"
25*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_deflate_predictor.h"
26*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_deflater.h"
27*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_frame.h"
28*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_inflater.h"
29*6777b538SAndroid Build Coastguard Worker #include "net/websockets/websocket_stream.h"
30*6777b538SAndroid Build Coastguard Worker 
31*6777b538SAndroid Build Coastguard Worker namespace net {
32*6777b538SAndroid Build Coastguard Worker class NetLogWithSource;
33*6777b538SAndroid Build Coastguard Worker 
34*6777b538SAndroid Build Coastguard Worker namespace {
35*6777b538SAndroid Build Coastguard Worker 
36*6777b538SAndroid Build Coastguard Worker constexpr int kWindowBits = 15;
37*6777b538SAndroid Build Coastguard Worker constexpr size_t kChunkSize = 4 * 1024;
38*6777b538SAndroid Build Coastguard Worker 
39*6777b538SAndroid Build Coastguard Worker }  // namespace
40*6777b538SAndroid Build Coastguard Worker 
WebSocketDeflateStream(std::unique_ptr<WebSocketStream> stream,const WebSocketDeflateParameters & params,std::unique_ptr<WebSocketDeflatePredictor> predictor)41*6777b538SAndroid Build Coastguard Worker WebSocketDeflateStream::WebSocketDeflateStream(
42*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<WebSocketStream> stream,
43*6777b538SAndroid Build Coastguard Worker     const WebSocketDeflateParameters& params,
44*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<WebSocketDeflatePredictor> predictor)
45*6777b538SAndroid Build Coastguard Worker     : stream_(std::move(stream)),
46*6777b538SAndroid Build Coastguard Worker       deflater_(params.client_context_take_over_mode()),
47*6777b538SAndroid Build Coastguard Worker       inflater_(kChunkSize, kChunkSize),
48*6777b538SAndroid Build Coastguard Worker       predictor_(std::move(predictor)) {
49*6777b538SAndroid Build Coastguard Worker   DCHECK(stream_);
50*6777b538SAndroid Build Coastguard Worker   DCHECK(params.IsValidAsResponse());
51*6777b538SAndroid Build Coastguard Worker   int client_max_window_bits = 15;
52*6777b538SAndroid Build Coastguard Worker   if (params.is_client_max_window_bits_specified()) {
53*6777b538SAndroid Build Coastguard Worker     DCHECK(params.has_client_max_window_bits_value());
54*6777b538SAndroid Build Coastguard Worker     client_max_window_bits = params.client_max_window_bits();
55*6777b538SAndroid Build Coastguard Worker   }
56*6777b538SAndroid Build Coastguard Worker   deflater_.Initialize(client_max_window_bits);
57*6777b538SAndroid Build Coastguard Worker   inflater_.Initialize(kWindowBits);
58*6777b538SAndroid Build Coastguard Worker }
59*6777b538SAndroid Build Coastguard Worker 
60*6777b538SAndroid Build Coastguard Worker WebSocketDeflateStream::~WebSocketDeflateStream() = default;
61*6777b538SAndroid Build Coastguard Worker 
ReadFrames(std::vector<std::unique_ptr<WebSocketFrame>> * frames,CompletionOnceCallback callback)62*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::ReadFrames(
63*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames,
64*6777b538SAndroid Build Coastguard Worker     CompletionOnceCallback callback) {
65*6777b538SAndroid Build Coastguard Worker   read_callback_ = std::move(callback);
66*6777b538SAndroid Build Coastguard Worker   inflater_outputs_.clear();
67*6777b538SAndroid Build Coastguard Worker   int result = stream_->ReadFrames(
68*6777b538SAndroid Build Coastguard Worker       frames, base::BindOnce(&WebSocketDeflateStream::OnReadComplete,
69*6777b538SAndroid Build Coastguard Worker                              base::Unretained(this), base::Unretained(frames)));
70*6777b538SAndroid Build Coastguard Worker   if (result < 0)
71*6777b538SAndroid Build Coastguard Worker     return result;
72*6777b538SAndroid Build Coastguard Worker   DCHECK_EQ(OK, result);
73*6777b538SAndroid Build Coastguard Worker   DCHECK(!frames->empty());
74*6777b538SAndroid Build Coastguard Worker 
75*6777b538SAndroid Build Coastguard Worker   return InflateAndReadIfNecessary(frames);
76*6777b538SAndroid Build Coastguard Worker }
77*6777b538SAndroid Build Coastguard Worker 
WriteFrames(std::vector<std::unique_ptr<WebSocketFrame>> * frames,CompletionOnceCallback callback)78*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::WriteFrames(
79*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames,
80*6777b538SAndroid Build Coastguard Worker     CompletionOnceCallback callback) {
81*6777b538SAndroid Build Coastguard Worker   deflater_outputs_.clear();
82*6777b538SAndroid Build Coastguard Worker   int result = Deflate(frames);
83*6777b538SAndroid Build Coastguard Worker   if (result != OK)
84*6777b538SAndroid Build Coastguard Worker     return result;
85*6777b538SAndroid Build Coastguard Worker   if (frames->empty())
86*6777b538SAndroid Build Coastguard Worker     return OK;
87*6777b538SAndroid Build Coastguard Worker   return stream_->WriteFrames(frames, std::move(callback));
88*6777b538SAndroid Build Coastguard Worker }
89*6777b538SAndroid Build Coastguard Worker 
Close()90*6777b538SAndroid Build Coastguard Worker void WebSocketDeflateStream::Close() { stream_->Close(); }
91*6777b538SAndroid Build Coastguard Worker 
GetSubProtocol() const92*6777b538SAndroid Build Coastguard Worker std::string WebSocketDeflateStream::GetSubProtocol() const {
93*6777b538SAndroid Build Coastguard Worker   return stream_->GetSubProtocol();
94*6777b538SAndroid Build Coastguard Worker }
95*6777b538SAndroid Build Coastguard Worker 
GetExtensions() const96*6777b538SAndroid Build Coastguard Worker std::string WebSocketDeflateStream::GetExtensions() const {
97*6777b538SAndroid Build Coastguard Worker   return stream_->GetExtensions();
98*6777b538SAndroid Build Coastguard Worker }
99*6777b538SAndroid Build Coastguard Worker 
GetNetLogWithSource() const100*6777b538SAndroid Build Coastguard Worker const NetLogWithSource& WebSocketDeflateStream::GetNetLogWithSource() const {
101*6777b538SAndroid Build Coastguard Worker   return stream_->GetNetLogWithSource();
102*6777b538SAndroid Build Coastguard Worker }
103*6777b538SAndroid Build Coastguard Worker 
OnReadComplete(std::vector<std::unique_ptr<WebSocketFrame>> * frames,int result)104*6777b538SAndroid Build Coastguard Worker void WebSocketDeflateStream::OnReadComplete(
105*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames,
106*6777b538SAndroid Build Coastguard Worker     int result) {
107*6777b538SAndroid Build Coastguard Worker   if (result != OK) {
108*6777b538SAndroid Build Coastguard Worker     frames->clear();
109*6777b538SAndroid Build Coastguard Worker     std::move(read_callback_).Run(result);
110*6777b538SAndroid Build Coastguard Worker     return;
111*6777b538SAndroid Build Coastguard Worker   }
112*6777b538SAndroid Build Coastguard Worker 
113*6777b538SAndroid Build Coastguard Worker   int r = InflateAndReadIfNecessary(frames);
114*6777b538SAndroid Build Coastguard Worker   if (r != ERR_IO_PENDING)
115*6777b538SAndroid Build Coastguard Worker     std::move(read_callback_).Run(r);
116*6777b538SAndroid Build Coastguard Worker }
117*6777b538SAndroid Build Coastguard Worker 
Deflate(std::vector<std::unique_ptr<WebSocketFrame>> * frames)118*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::Deflate(
119*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames) {
120*6777b538SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<WebSocketFrame>> frames_to_write;
121*6777b538SAndroid Build Coastguard Worker   // Store frames of the currently processed message if writing_state_ equals to
122*6777b538SAndroid Build Coastguard Worker   // WRITING_POSSIBLY_COMPRESSED_MESSAGE.
123*6777b538SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<WebSocketFrame>> frames_of_message;
124*6777b538SAndroid Build Coastguard Worker   for (size_t i = 0; i < frames->size(); ++i) {
125*6777b538SAndroid Build Coastguard Worker     DCHECK(!(*frames)[i]->header.reserved1);
126*6777b538SAndroid Build Coastguard Worker     if (!WebSocketFrameHeader::IsKnownDataOpCode((*frames)[i]->header.opcode)) {
127*6777b538SAndroid Build Coastguard Worker       frames_to_write.push_back(std::move((*frames)[i]));
128*6777b538SAndroid Build Coastguard Worker       continue;
129*6777b538SAndroid Build Coastguard Worker     }
130*6777b538SAndroid Build Coastguard Worker     if (writing_state_ == NOT_WRITING)
131*6777b538SAndroid Build Coastguard Worker       OnMessageStart(*frames, i);
132*6777b538SAndroid Build Coastguard Worker 
133*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<WebSocketFrame> frame(std::move((*frames)[i]));
134*6777b538SAndroid Build Coastguard Worker     predictor_->RecordInputDataFrame(frame.get());
135*6777b538SAndroid Build Coastguard Worker 
136*6777b538SAndroid Build Coastguard Worker     if (writing_state_ == WRITING_UNCOMPRESSED_MESSAGE) {
137*6777b538SAndroid Build Coastguard Worker       if (frame->header.final)
138*6777b538SAndroid Build Coastguard Worker         writing_state_ = NOT_WRITING;
139*6777b538SAndroid Build Coastguard Worker       predictor_->RecordWrittenDataFrame(frame.get());
140*6777b538SAndroid Build Coastguard Worker       frames_to_write.push_back(std::move(frame));
141*6777b538SAndroid Build Coastguard Worker       current_writing_opcode_ = WebSocketFrameHeader::kOpCodeContinuation;
142*6777b538SAndroid Build Coastguard Worker     } else {
143*6777b538SAndroid Build Coastguard Worker       if (frame->payload &&
144*6777b538SAndroid Build Coastguard Worker           !deflater_.AddBytes(
145*6777b538SAndroid Build Coastguard Worker               frame->payload,
146*6777b538SAndroid Build Coastguard Worker               static_cast<size_t>(frame->header.payload_length))) {
147*6777b538SAndroid Build Coastguard Worker         DVLOG(1) << "WebSocket protocol error. "
148*6777b538SAndroid Build Coastguard Worker                  << "deflater_.AddBytes() returns an error.";
149*6777b538SAndroid Build Coastguard Worker         return ERR_WS_PROTOCOL_ERROR;
150*6777b538SAndroid Build Coastguard Worker       }
151*6777b538SAndroid Build Coastguard Worker       if (frame->header.final && !deflater_.Finish()) {
152*6777b538SAndroid Build Coastguard Worker         DVLOG(1) << "WebSocket protocol error. "
153*6777b538SAndroid Build Coastguard Worker                  << "deflater_.Finish() returns an error.";
154*6777b538SAndroid Build Coastguard Worker         return ERR_WS_PROTOCOL_ERROR;
155*6777b538SAndroid Build Coastguard Worker       }
156*6777b538SAndroid Build Coastguard Worker 
157*6777b538SAndroid Build Coastguard Worker       if (writing_state_ == WRITING_COMPRESSED_MESSAGE) {
158*6777b538SAndroid Build Coastguard Worker         if (deflater_.CurrentOutputSize() >= kChunkSize ||
159*6777b538SAndroid Build Coastguard Worker             frame->header.final) {
160*6777b538SAndroid Build Coastguard Worker           int result = AppendCompressedFrame(frame->header, &frames_to_write);
161*6777b538SAndroid Build Coastguard Worker           if (result != OK)
162*6777b538SAndroid Build Coastguard Worker             return result;
163*6777b538SAndroid Build Coastguard Worker         }
164*6777b538SAndroid Build Coastguard Worker         if (frame->header.final)
165*6777b538SAndroid Build Coastguard Worker           writing_state_ = NOT_WRITING;
166*6777b538SAndroid Build Coastguard Worker       } else {
167*6777b538SAndroid Build Coastguard Worker         DCHECK_EQ(WRITING_POSSIBLY_COMPRESSED_MESSAGE, writing_state_);
168*6777b538SAndroid Build Coastguard Worker         bool final = frame->header.final;
169*6777b538SAndroid Build Coastguard Worker         frames_of_message.push_back(std::move(frame));
170*6777b538SAndroid Build Coastguard Worker         if (final) {
171*6777b538SAndroid Build Coastguard Worker           int result = AppendPossiblyCompressedMessage(&frames_of_message,
172*6777b538SAndroid Build Coastguard Worker                                                        &frames_to_write);
173*6777b538SAndroid Build Coastguard Worker           if (result != OK)
174*6777b538SAndroid Build Coastguard Worker             return result;
175*6777b538SAndroid Build Coastguard Worker           frames_of_message.clear();
176*6777b538SAndroid Build Coastguard Worker           writing_state_ = NOT_WRITING;
177*6777b538SAndroid Build Coastguard Worker         }
178*6777b538SAndroid Build Coastguard Worker       }
179*6777b538SAndroid Build Coastguard Worker     }
180*6777b538SAndroid Build Coastguard Worker   }
181*6777b538SAndroid Build Coastguard Worker   DCHECK_NE(WRITING_POSSIBLY_COMPRESSED_MESSAGE, writing_state_);
182*6777b538SAndroid Build Coastguard Worker   frames->swap(frames_to_write);
183*6777b538SAndroid Build Coastguard Worker   return OK;
184*6777b538SAndroid Build Coastguard Worker }
185*6777b538SAndroid Build Coastguard Worker 
OnMessageStart(const std::vector<std::unique_ptr<WebSocketFrame>> & frames,size_t index)186*6777b538SAndroid Build Coastguard Worker void WebSocketDeflateStream::OnMessageStart(
187*6777b538SAndroid Build Coastguard Worker     const std::vector<std::unique_ptr<WebSocketFrame>>& frames,
188*6777b538SAndroid Build Coastguard Worker     size_t index) {
189*6777b538SAndroid Build Coastguard Worker   WebSocketFrame* frame = frames[index].get();
190*6777b538SAndroid Build Coastguard Worker   current_writing_opcode_ = frame->header.opcode;
191*6777b538SAndroid Build Coastguard Worker   DCHECK(current_writing_opcode_ == WebSocketFrameHeader::kOpCodeText ||
192*6777b538SAndroid Build Coastguard Worker          current_writing_opcode_ == WebSocketFrameHeader::kOpCodeBinary);
193*6777b538SAndroid Build Coastguard Worker   WebSocketDeflatePredictor::Result prediction =
194*6777b538SAndroid Build Coastguard Worker       predictor_->Predict(frames, index);
195*6777b538SAndroid Build Coastguard Worker 
196*6777b538SAndroid Build Coastguard Worker   switch (prediction) {
197*6777b538SAndroid Build Coastguard Worker     case WebSocketDeflatePredictor::DEFLATE:
198*6777b538SAndroid Build Coastguard Worker       writing_state_ = WRITING_COMPRESSED_MESSAGE;
199*6777b538SAndroid Build Coastguard Worker       return;
200*6777b538SAndroid Build Coastguard Worker     case WebSocketDeflatePredictor::DO_NOT_DEFLATE:
201*6777b538SAndroid Build Coastguard Worker       writing_state_ = WRITING_UNCOMPRESSED_MESSAGE;
202*6777b538SAndroid Build Coastguard Worker       return;
203*6777b538SAndroid Build Coastguard Worker     case WebSocketDeflatePredictor::TRY_DEFLATE:
204*6777b538SAndroid Build Coastguard Worker       writing_state_ = WRITING_POSSIBLY_COMPRESSED_MESSAGE;
205*6777b538SAndroid Build Coastguard Worker       return;
206*6777b538SAndroid Build Coastguard Worker   }
207*6777b538SAndroid Build Coastguard Worker   NOTREACHED();
208*6777b538SAndroid Build Coastguard Worker }
209*6777b538SAndroid Build Coastguard Worker 
AppendCompressedFrame(const WebSocketFrameHeader & header,std::vector<std::unique_ptr<WebSocketFrame>> * frames_to_write)210*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::AppendCompressedFrame(
211*6777b538SAndroid Build Coastguard Worker     const WebSocketFrameHeader& header,
212*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames_to_write) {
213*6777b538SAndroid Build Coastguard Worker   const WebSocketFrameHeader::OpCode opcode = current_writing_opcode_;
214*6777b538SAndroid Build Coastguard Worker   scoped_refptr<IOBufferWithSize> compressed_payload =
215*6777b538SAndroid Build Coastguard Worker       deflater_.GetOutput(deflater_.CurrentOutputSize());
216*6777b538SAndroid Build Coastguard Worker   if (!compressed_payload.get()) {
217*6777b538SAndroid Build Coastguard Worker     DVLOG(1) << "WebSocket protocol error. "
218*6777b538SAndroid Build Coastguard Worker              << "deflater_.GetOutput() returns an error.";
219*6777b538SAndroid Build Coastguard Worker     return ERR_WS_PROTOCOL_ERROR;
220*6777b538SAndroid Build Coastguard Worker   }
221*6777b538SAndroid Build Coastguard Worker   deflater_outputs_.push_back(compressed_payload);
222*6777b538SAndroid Build Coastguard Worker   auto compressed = std::make_unique<WebSocketFrame>(opcode);
223*6777b538SAndroid Build Coastguard Worker   compressed->header.CopyFrom(header);
224*6777b538SAndroid Build Coastguard Worker   compressed->header.opcode = opcode;
225*6777b538SAndroid Build Coastguard Worker   compressed->header.final = header.final;
226*6777b538SAndroid Build Coastguard Worker   compressed->header.reserved1 =
227*6777b538SAndroid Build Coastguard Worker       (opcode != WebSocketFrameHeader::kOpCodeContinuation);
228*6777b538SAndroid Build Coastguard Worker   compressed->payload = compressed_payload->data();
229*6777b538SAndroid Build Coastguard Worker   compressed->header.payload_length = compressed_payload->size();
230*6777b538SAndroid Build Coastguard Worker 
231*6777b538SAndroid Build Coastguard Worker   current_writing_opcode_ = WebSocketFrameHeader::kOpCodeContinuation;
232*6777b538SAndroid Build Coastguard Worker   predictor_->RecordWrittenDataFrame(compressed.get());
233*6777b538SAndroid Build Coastguard Worker   frames_to_write->push_back(std::move(compressed));
234*6777b538SAndroid Build Coastguard Worker   return OK;
235*6777b538SAndroid Build Coastguard Worker }
236*6777b538SAndroid Build Coastguard Worker 
AppendPossiblyCompressedMessage(std::vector<std::unique_ptr<WebSocketFrame>> * frames,std::vector<std::unique_ptr<WebSocketFrame>> * frames_to_write)237*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::AppendPossiblyCompressedMessage(
238*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames,
239*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames_to_write) {
240*6777b538SAndroid Build Coastguard Worker   DCHECK(!frames->empty());
241*6777b538SAndroid Build Coastguard Worker 
242*6777b538SAndroid Build Coastguard Worker   const WebSocketFrameHeader::OpCode opcode = current_writing_opcode_;
243*6777b538SAndroid Build Coastguard Worker   scoped_refptr<IOBufferWithSize> compressed_payload =
244*6777b538SAndroid Build Coastguard Worker       deflater_.GetOutput(deflater_.CurrentOutputSize());
245*6777b538SAndroid Build Coastguard Worker   if (!compressed_payload.get()) {
246*6777b538SAndroid Build Coastguard Worker     DVLOG(1) << "WebSocket protocol error. "
247*6777b538SAndroid Build Coastguard Worker              << "deflater_.GetOutput() returns an error.";
248*6777b538SAndroid Build Coastguard Worker     return ERR_WS_PROTOCOL_ERROR;
249*6777b538SAndroid Build Coastguard Worker   }
250*6777b538SAndroid Build Coastguard Worker   deflater_outputs_.push_back(compressed_payload);
251*6777b538SAndroid Build Coastguard Worker 
252*6777b538SAndroid Build Coastguard Worker   uint64_t original_payload_length = 0;
253*6777b538SAndroid Build Coastguard Worker   for (size_t i = 0; i < frames->size(); ++i) {
254*6777b538SAndroid Build Coastguard Worker     WebSocketFrame* frame = (*frames)[i].get();
255*6777b538SAndroid Build Coastguard Worker     // Asserts checking that frames represent one whole data message.
256*6777b538SAndroid Build Coastguard Worker     DCHECK(WebSocketFrameHeader::IsKnownDataOpCode(frame->header.opcode));
257*6777b538SAndroid Build Coastguard Worker     DCHECK_EQ(i == 0,
258*6777b538SAndroid Build Coastguard Worker               WebSocketFrameHeader::kOpCodeContinuation !=
259*6777b538SAndroid Build Coastguard Worker               frame->header.opcode);
260*6777b538SAndroid Build Coastguard Worker     DCHECK_EQ(i == frames->size() - 1, frame->header.final);
261*6777b538SAndroid Build Coastguard Worker     original_payload_length += frame->header.payload_length;
262*6777b538SAndroid Build Coastguard Worker   }
263*6777b538SAndroid Build Coastguard Worker   if (original_payload_length <=
264*6777b538SAndroid Build Coastguard Worker       static_cast<uint64_t>(compressed_payload->size())) {
265*6777b538SAndroid Build Coastguard Worker     // Compression is not effective. Use the original frames.
266*6777b538SAndroid Build Coastguard Worker     for (auto& frame : *frames) {
267*6777b538SAndroid Build Coastguard Worker       predictor_->RecordWrittenDataFrame(frame.get());
268*6777b538SAndroid Build Coastguard Worker       frames_to_write->push_back(std::move(frame));
269*6777b538SAndroid Build Coastguard Worker     }
270*6777b538SAndroid Build Coastguard Worker     frames->clear();
271*6777b538SAndroid Build Coastguard Worker     return OK;
272*6777b538SAndroid Build Coastguard Worker   }
273*6777b538SAndroid Build Coastguard Worker   auto compressed = std::make_unique<WebSocketFrame>(opcode);
274*6777b538SAndroid Build Coastguard Worker   compressed->header.CopyFrom((*frames)[0]->header);
275*6777b538SAndroid Build Coastguard Worker   compressed->header.opcode = opcode;
276*6777b538SAndroid Build Coastguard Worker   compressed->header.final = true;
277*6777b538SAndroid Build Coastguard Worker   compressed->header.reserved1 = true;
278*6777b538SAndroid Build Coastguard Worker   compressed->payload = compressed_payload->data();
279*6777b538SAndroid Build Coastguard Worker   compressed->header.payload_length = compressed_payload->size();
280*6777b538SAndroid Build Coastguard Worker 
281*6777b538SAndroid Build Coastguard Worker   predictor_->RecordWrittenDataFrame(compressed.get());
282*6777b538SAndroid Build Coastguard Worker   frames_to_write->push_back(std::move(compressed));
283*6777b538SAndroid Build Coastguard Worker   return OK;
284*6777b538SAndroid Build Coastguard Worker }
285*6777b538SAndroid Build Coastguard Worker 
Inflate(std::vector<std::unique_ptr<WebSocketFrame>> * frames)286*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::Inflate(
287*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames) {
288*6777b538SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<WebSocketFrame>> frames_to_output;
289*6777b538SAndroid Build Coastguard Worker   std::vector<std::unique_ptr<WebSocketFrame>> frames_passed;
290*6777b538SAndroid Build Coastguard Worker   frames->swap(frames_passed);
291*6777b538SAndroid Build Coastguard Worker   for (auto& frame_passed : frames_passed) {
292*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<WebSocketFrame> frame(std::move(frame_passed));
293*6777b538SAndroid Build Coastguard Worker     frame_passed = nullptr;
294*6777b538SAndroid Build Coastguard Worker     DVLOG(3) << "Input frame: opcode=" << frame->header.opcode
295*6777b538SAndroid Build Coastguard Worker              << " final=" << frame->header.final
296*6777b538SAndroid Build Coastguard Worker              << " reserved1=" << frame->header.reserved1
297*6777b538SAndroid Build Coastguard Worker              << " payload_length=" << frame->header.payload_length;
298*6777b538SAndroid Build Coastguard Worker 
299*6777b538SAndroid Build Coastguard Worker     if (!WebSocketFrameHeader::IsKnownDataOpCode(frame->header.opcode)) {
300*6777b538SAndroid Build Coastguard Worker       frames_to_output.push_back(std::move(frame));
301*6777b538SAndroid Build Coastguard Worker       continue;
302*6777b538SAndroid Build Coastguard Worker     }
303*6777b538SAndroid Build Coastguard Worker 
304*6777b538SAndroid Build Coastguard Worker     if (reading_state_ == NOT_READING) {
305*6777b538SAndroid Build Coastguard Worker       if (frame->header.reserved1)
306*6777b538SAndroid Build Coastguard Worker         reading_state_ = READING_COMPRESSED_MESSAGE;
307*6777b538SAndroid Build Coastguard Worker       else
308*6777b538SAndroid Build Coastguard Worker         reading_state_ = READING_UNCOMPRESSED_MESSAGE;
309*6777b538SAndroid Build Coastguard Worker       current_reading_opcode_ = frame->header.opcode;
310*6777b538SAndroid Build Coastguard Worker     } else {
311*6777b538SAndroid Build Coastguard Worker       if (frame->header.reserved1) {
312*6777b538SAndroid Build Coastguard Worker         DVLOG(1) << "WebSocket protocol error. "
313*6777b538SAndroid Build Coastguard Worker                  << "Receiving a non-first frame with RSV1 flag set.";
314*6777b538SAndroid Build Coastguard Worker         return ERR_WS_PROTOCOL_ERROR;
315*6777b538SAndroid Build Coastguard Worker       }
316*6777b538SAndroid Build Coastguard Worker     }
317*6777b538SAndroid Build Coastguard Worker 
318*6777b538SAndroid Build Coastguard Worker     if (reading_state_ == READING_UNCOMPRESSED_MESSAGE) {
319*6777b538SAndroid Build Coastguard Worker       if (frame->header.final)
320*6777b538SAndroid Build Coastguard Worker         reading_state_ = NOT_READING;
321*6777b538SAndroid Build Coastguard Worker       current_reading_opcode_ = WebSocketFrameHeader::kOpCodeContinuation;
322*6777b538SAndroid Build Coastguard Worker       frames_to_output.push_back(std::move(frame));
323*6777b538SAndroid Build Coastguard Worker     } else {
324*6777b538SAndroid Build Coastguard Worker       DCHECK_EQ(reading_state_, READING_COMPRESSED_MESSAGE);
325*6777b538SAndroid Build Coastguard Worker       if (frame->payload &&
326*6777b538SAndroid Build Coastguard Worker           !inflater_.AddBytes(
327*6777b538SAndroid Build Coastguard Worker               frame->payload,
328*6777b538SAndroid Build Coastguard Worker               static_cast<size_t>(frame->header.payload_length))) {
329*6777b538SAndroid Build Coastguard Worker         DVLOG(1) << "WebSocket protocol error. "
330*6777b538SAndroid Build Coastguard Worker                  << "inflater_.AddBytes() returns an error.";
331*6777b538SAndroid Build Coastguard Worker         return ERR_WS_PROTOCOL_ERROR;
332*6777b538SAndroid Build Coastguard Worker       }
333*6777b538SAndroid Build Coastguard Worker       if (frame->header.final) {
334*6777b538SAndroid Build Coastguard Worker         if (!inflater_.Finish()) {
335*6777b538SAndroid Build Coastguard Worker           DVLOG(1) << "WebSocket protocol error. "
336*6777b538SAndroid Build Coastguard Worker                    << "inflater_.Finish() returns an error.";
337*6777b538SAndroid Build Coastguard Worker           return ERR_WS_PROTOCOL_ERROR;
338*6777b538SAndroid Build Coastguard Worker         }
339*6777b538SAndroid Build Coastguard Worker       }
340*6777b538SAndroid Build Coastguard Worker       // TODO(yhirano): Many frames can be generated by the inflater and
341*6777b538SAndroid Build Coastguard Worker       // memory consumption can grow.
342*6777b538SAndroid Build Coastguard Worker       // We could avoid it, but avoiding it makes this class much more
343*6777b538SAndroid Build Coastguard Worker       // complicated.
344*6777b538SAndroid Build Coastguard Worker       while (inflater_.CurrentOutputSize() >= kChunkSize ||
345*6777b538SAndroid Build Coastguard Worker              frame->header.final) {
346*6777b538SAndroid Build Coastguard Worker         size_t size = std::min(kChunkSize, inflater_.CurrentOutputSize());
347*6777b538SAndroid Build Coastguard Worker         auto inflated =
348*6777b538SAndroid Build Coastguard Worker             std::make_unique<WebSocketFrame>(WebSocketFrameHeader::kOpCodeText);
349*6777b538SAndroid Build Coastguard Worker         scoped_refptr<IOBufferWithSize> data = inflater_.GetOutput(size);
350*6777b538SAndroid Build Coastguard Worker         inflater_outputs_.push_back(data);
351*6777b538SAndroid Build Coastguard Worker         bool is_final = !inflater_.CurrentOutputSize() && frame->header.final;
352*6777b538SAndroid Build Coastguard Worker         if (!data.get()) {
353*6777b538SAndroid Build Coastguard Worker           DVLOG(1) << "WebSocket protocol error. "
354*6777b538SAndroid Build Coastguard Worker                    << "inflater_.GetOutput() returns an error.";
355*6777b538SAndroid Build Coastguard Worker           return ERR_WS_PROTOCOL_ERROR;
356*6777b538SAndroid Build Coastguard Worker         }
357*6777b538SAndroid Build Coastguard Worker         inflated->header.CopyFrom(frame->header);
358*6777b538SAndroid Build Coastguard Worker         inflated->header.opcode = current_reading_opcode_;
359*6777b538SAndroid Build Coastguard Worker         inflated->header.final = is_final;
360*6777b538SAndroid Build Coastguard Worker         inflated->header.reserved1 = false;
361*6777b538SAndroid Build Coastguard Worker         inflated->payload = data->data();
362*6777b538SAndroid Build Coastguard Worker         inflated->header.payload_length = data->size();
363*6777b538SAndroid Build Coastguard Worker         DVLOG(3) << "Inflated frame: opcode=" << inflated->header.opcode
364*6777b538SAndroid Build Coastguard Worker                  << " final=" << inflated->header.final
365*6777b538SAndroid Build Coastguard Worker                  << " reserved1=" << inflated->header.reserved1
366*6777b538SAndroid Build Coastguard Worker                  << " payload_length=" << inflated->header.payload_length;
367*6777b538SAndroid Build Coastguard Worker         frames_to_output.push_back(std::move(inflated));
368*6777b538SAndroid Build Coastguard Worker         current_reading_opcode_ = WebSocketFrameHeader::kOpCodeContinuation;
369*6777b538SAndroid Build Coastguard Worker         if (is_final)
370*6777b538SAndroid Build Coastguard Worker           break;
371*6777b538SAndroid Build Coastguard Worker       }
372*6777b538SAndroid Build Coastguard Worker       if (frame->header.final)
373*6777b538SAndroid Build Coastguard Worker         reading_state_ = NOT_READING;
374*6777b538SAndroid Build Coastguard Worker     }
375*6777b538SAndroid Build Coastguard Worker   }
376*6777b538SAndroid Build Coastguard Worker   frames->swap(frames_to_output);
377*6777b538SAndroid Build Coastguard Worker   return frames->empty() ? ERR_IO_PENDING : OK;
378*6777b538SAndroid Build Coastguard Worker }
379*6777b538SAndroid Build Coastguard Worker 
InflateAndReadIfNecessary(std::vector<std::unique_ptr<WebSocketFrame>> * frames)380*6777b538SAndroid Build Coastguard Worker int WebSocketDeflateStream::InflateAndReadIfNecessary(
381*6777b538SAndroid Build Coastguard Worker     std::vector<std::unique_ptr<WebSocketFrame>>* frames) {
382*6777b538SAndroid Build Coastguard Worker   int result = Inflate(frames);
383*6777b538SAndroid Build Coastguard Worker   while (result == ERR_IO_PENDING) {
384*6777b538SAndroid Build Coastguard Worker     DCHECK(frames->empty());
385*6777b538SAndroid Build Coastguard Worker 
386*6777b538SAndroid Build Coastguard Worker     result = stream_->ReadFrames(
387*6777b538SAndroid Build Coastguard Worker         frames,
388*6777b538SAndroid Build Coastguard Worker         base::BindOnce(&WebSocketDeflateStream::OnReadComplete,
389*6777b538SAndroid Build Coastguard Worker                        base::Unretained(this), base::Unretained(frames)));
390*6777b538SAndroid Build Coastguard Worker     if (result < 0)
391*6777b538SAndroid Build Coastguard Worker       break;
392*6777b538SAndroid Build Coastguard Worker     DCHECK_EQ(OK, result);
393*6777b538SAndroid Build Coastguard Worker     DCHECK(!frames->empty());
394*6777b538SAndroid Build Coastguard Worker 
395*6777b538SAndroid Build Coastguard Worker     result = Inflate(frames);
396*6777b538SAndroid Build Coastguard Worker   }
397*6777b538SAndroid Build Coastguard Worker   if (result < 0)
398*6777b538SAndroid Build Coastguard Worker     frames->clear();
399*6777b538SAndroid Build Coastguard Worker   return result;
400*6777b538SAndroid Build Coastguard Worker }
401*6777b538SAndroid Build Coastguard Worker 
402*6777b538SAndroid Build Coastguard Worker }  // namespace net
403