xref: /aosp_15_r20/external/cronet/net/server/http_connection.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 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/server/http_connection.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <utility>
8*6777b538SAndroid Build Coastguard Worker 
9*6777b538SAndroid Build Coastguard Worker #include "base/logging.h"
10*6777b538SAndroid Build Coastguard Worker #include "net/server/web_socket.h"
11*6777b538SAndroid Build Coastguard Worker #include "net/socket/stream_socket.h"
12*6777b538SAndroid Build Coastguard Worker 
13*6777b538SAndroid Build Coastguard Worker namespace net {
14*6777b538SAndroid Build Coastguard Worker 
ReadIOBuffer()15*6777b538SAndroid Build Coastguard Worker HttpConnection::ReadIOBuffer::ReadIOBuffer()
16*6777b538SAndroid Build Coastguard Worker     : base_(base::MakeRefCounted<GrowableIOBuffer>()) {
17*6777b538SAndroid Build Coastguard Worker   SetCapacity(kInitialBufSize);
18*6777b538SAndroid Build Coastguard Worker }
19*6777b538SAndroid Build Coastguard Worker 
~ReadIOBuffer()20*6777b538SAndroid Build Coastguard Worker HttpConnection::ReadIOBuffer::~ReadIOBuffer() {
21*6777b538SAndroid Build Coastguard Worker   data_ = nullptr;  // Avoid dangling ptr when `base_` is destroyed.
22*6777b538SAndroid Build Coastguard Worker }
23*6777b538SAndroid Build Coastguard Worker 
GetCapacity() const24*6777b538SAndroid Build Coastguard Worker int HttpConnection::ReadIOBuffer::GetCapacity() const {
25*6777b538SAndroid Build Coastguard Worker   return base_->capacity();
26*6777b538SAndroid Build Coastguard Worker }
27*6777b538SAndroid Build Coastguard Worker 
SetCapacity(int capacity)28*6777b538SAndroid Build Coastguard Worker void HttpConnection::ReadIOBuffer::SetCapacity(int capacity) {
29*6777b538SAndroid Build Coastguard Worker   DCHECK_LE(GetSize(), capacity);
30*6777b538SAndroid Build Coastguard Worker   data_ = nullptr;
31*6777b538SAndroid Build Coastguard Worker   base_->SetCapacity(capacity);
32*6777b538SAndroid Build Coastguard Worker   data_ = base_->data();
33*6777b538SAndroid Build Coastguard Worker }
34*6777b538SAndroid Build Coastguard Worker 
IncreaseCapacity()35*6777b538SAndroid Build Coastguard Worker bool HttpConnection::ReadIOBuffer::IncreaseCapacity() {
36*6777b538SAndroid Build Coastguard Worker   if (GetCapacity() >= max_buffer_size_) {
37*6777b538SAndroid Build Coastguard Worker     LOG(ERROR) << "Too large read data is pending: capacity=" << GetCapacity()
38*6777b538SAndroid Build Coastguard Worker                << ", max_buffer_size=" << max_buffer_size_
39*6777b538SAndroid Build Coastguard Worker                << ", read=" << GetSize();
40*6777b538SAndroid Build Coastguard Worker     return false;
41*6777b538SAndroid Build Coastguard Worker   }
42*6777b538SAndroid Build Coastguard Worker 
43*6777b538SAndroid Build Coastguard Worker   int new_capacity = GetCapacity() * kCapacityIncreaseFactor;
44*6777b538SAndroid Build Coastguard Worker   if (new_capacity > max_buffer_size_)
45*6777b538SAndroid Build Coastguard Worker     new_capacity = max_buffer_size_;
46*6777b538SAndroid Build Coastguard Worker   SetCapacity(new_capacity);
47*6777b538SAndroid Build Coastguard Worker   return true;
48*6777b538SAndroid Build Coastguard Worker }
49*6777b538SAndroid Build Coastguard Worker 
StartOfBuffer() const50*6777b538SAndroid Build Coastguard Worker char* HttpConnection::ReadIOBuffer::StartOfBuffer() const {
51*6777b538SAndroid Build Coastguard Worker   return base_->StartOfBuffer();
52*6777b538SAndroid Build Coastguard Worker }
53*6777b538SAndroid Build Coastguard Worker 
GetSize() const54*6777b538SAndroid Build Coastguard Worker int HttpConnection::ReadIOBuffer::GetSize() const {
55*6777b538SAndroid Build Coastguard Worker   return base_->offset();
56*6777b538SAndroid Build Coastguard Worker }
57*6777b538SAndroid Build Coastguard Worker 
DidRead(int bytes)58*6777b538SAndroid Build Coastguard Worker void HttpConnection::ReadIOBuffer::DidRead(int bytes) {
59*6777b538SAndroid Build Coastguard Worker   DCHECK_GE(RemainingCapacity(), bytes);
60*6777b538SAndroid Build Coastguard Worker   base_->set_offset(base_->offset() + bytes);
61*6777b538SAndroid Build Coastguard Worker   data_ = base_->data();
62*6777b538SAndroid Build Coastguard Worker }
63*6777b538SAndroid Build Coastguard Worker 
RemainingCapacity() const64*6777b538SAndroid Build Coastguard Worker int HttpConnection::ReadIOBuffer::RemainingCapacity() const {
65*6777b538SAndroid Build Coastguard Worker   return base_->RemainingCapacity();
66*6777b538SAndroid Build Coastguard Worker }
67*6777b538SAndroid Build Coastguard Worker 
DidConsume(int bytes)68*6777b538SAndroid Build Coastguard Worker void HttpConnection::ReadIOBuffer::DidConsume(int bytes) {
69*6777b538SAndroid Build Coastguard Worker   int previous_size = GetSize();
70*6777b538SAndroid Build Coastguard Worker   int unconsumed_size = previous_size - bytes;
71*6777b538SAndroid Build Coastguard Worker   DCHECK_LE(0, unconsumed_size);
72*6777b538SAndroid Build Coastguard Worker   if (unconsumed_size > 0) {
73*6777b538SAndroid Build Coastguard Worker     // Move unconsumed data to the start of buffer.
74*6777b538SAndroid Build Coastguard Worker     memmove(StartOfBuffer(), StartOfBuffer() + bytes, unconsumed_size);
75*6777b538SAndroid Build Coastguard Worker   }
76*6777b538SAndroid Build Coastguard Worker   base_->set_offset(unconsumed_size);
77*6777b538SAndroid Build Coastguard Worker   data_ = base_->data();
78*6777b538SAndroid Build Coastguard Worker 
79*6777b538SAndroid Build Coastguard Worker   // If capacity is too big, reduce it.
80*6777b538SAndroid Build Coastguard Worker   if (GetCapacity() > kMinimumBufSize &&
81*6777b538SAndroid Build Coastguard Worker       GetCapacity() > previous_size * kCapacityIncreaseFactor) {
82*6777b538SAndroid Build Coastguard Worker     int new_capacity = GetCapacity() / kCapacityIncreaseFactor;
83*6777b538SAndroid Build Coastguard Worker     if (new_capacity < kMinimumBufSize)
84*6777b538SAndroid Build Coastguard Worker       new_capacity = kMinimumBufSize;
85*6777b538SAndroid Build Coastguard Worker     // this avoids the pointer to dangle until `SetCapacity` gets called.
86*6777b538SAndroid Build Coastguard Worker     data_ = nullptr;
87*6777b538SAndroid Build Coastguard Worker     // realloc() within GrowableIOBuffer::SetCapacity() could move data even
88*6777b538SAndroid Build Coastguard Worker     // when size is reduced. If unconsumed_size == 0, i.e. no data exists in
89*6777b538SAndroid Build Coastguard Worker     // the buffer, free internal buffer first to guarantee no data move.
90*6777b538SAndroid Build Coastguard Worker     if (!unconsumed_size)
91*6777b538SAndroid Build Coastguard Worker       base_->SetCapacity(0);
92*6777b538SAndroid Build Coastguard Worker     SetCapacity(new_capacity);
93*6777b538SAndroid Build Coastguard Worker   }
94*6777b538SAndroid Build Coastguard Worker }
95*6777b538SAndroid Build Coastguard Worker 
96*6777b538SAndroid Build Coastguard Worker HttpConnection::QueuedWriteIOBuffer::QueuedWriteIOBuffer() = default;
97*6777b538SAndroid Build Coastguard Worker 
~QueuedWriteIOBuffer()98*6777b538SAndroid Build Coastguard Worker HttpConnection::QueuedWriteIOBuffer::~QueuedWriteIOBuffer() {
99*6777b538SAndroid Build Coastguard Worker   data_ = nullptr;  // pending_data_ owns data_.
100*6777b538SAndroid Build Coastguard Worker }
101*6777b538SAndroid Build Coastguard Worker 
IsEmpty() const102*6777b538SAndroid Build Coastguard Worker bool HttpConnection::QueuedWriteIOBuffer::IsEmpty() const {
103*6777b538SAndroid Build Coastguard Worker   return pending_data_.empty();
104*6777b538SAndroid Build Coastguard Worker }
105*6777b538SAndroid Build Coastguard Worker 
Append(const std::string & data)106*6777b538SAndroid Build Coastguard Worker bool HttpConnection::QueuedWriteIOBuffer::Append(const std::string& data) {
107*6777b538SAndroid Build Coastguard Worker   if (data.empty())
108*6777b538SAndroid Build Coastguard Worker     return true;
109*6777b538SAndroid Build Coastguard Worker 
110*6777b538SAndroid Build Coastguard Worker   if (total_size_ + static_cast<int>(data.size()) > max_buffer_size_) {
111*6777b538SAndroid Build Coastguard Worker     LOG(ERROR) << "Too large write data is pending: size="
112*6777b538SAndroid Build Coastguard Worker                << total_size_ + data.size()
113*6777b538SAndroid Build Coastguard Worker                << ", max_buffer_size=" << max_buffer_size_;
114*6777b538SAndroid Build Coastguard Worker     return false;
115*6777b538SAndroid Build Coastguard Worker   }
116*6777b538SAndroid Build Coastguard Worker 
117*6777b538SAndroid Build Coastguard Worker   pending_data_.push(std::make_unique<std::string>(data));
118*6777b538SAndroid Build Coastguard Worker   total_size_ += data.size();
119*6777b538SAndroid Build Coastguard Worker 
120*6777b538SAndroid Build Coastguard Worker   // If new data is the first pending data, updates data_.
121*6777b538SAndroid Build Coastguard Worker   if (pending_data_.size() == 1)
122*6777b538SAndroid Build Coastguard Worker     data_ = const_cast<char*>(pending_data_.front()->data());
123*6777b538SAndroid Build Coastguard Worker   return true;
124*6777b538SAndroid Build Coastguard Worker }
125*6777b538SAndroid Build Coastguard Worker 
DidConsume(int size)126*6777b538SAndroid Build Coastguard Worker void HttpConnection::QueuedWriteIOBuffer::DidConsume(int size) {
127*6777b538SAndroid Build Coastguard Worker   DCHECK_GE(total_size_, size);
128*6777b538SAndroid Build Coastguard Worker   DCHECK_GE(GetSizeToWrite(), size);
129*6777b538SAndroid Build Coastguard Worker   if (size == 0)
130*6777b538SAndroid Build Coastguard Worker     return;
131*6777b538SAndroid Build Coastguard Worker 
132*6777b538SAndroid Build Coastguard Worker   if (size < GetSizeToWrite()) {
133*6777b538SAndroid Build Coastguard Worker     data_ += size;
134*6777b538SAndroid Build Coastguard Worker   } else {  // size == GetSizeToWrite(). Updates data_ to next pending data.
135*6777b538SAndroid Build Coastguard Worker     data_ = nullptr;
136*6777b538SAndroid Build Coastguard Worker     pending_data_.pop();
137*6777b538SAndroid Build Coastguard Worker     data_ =
138*6777b538SAndroid Build Coastguard Worker         IsEmpty() ? nullptr : const_cast<char*>(pending_data_.front()->data());
139*6777b538SAndroid Build Coastguard Worker   }
140*6777b538SAndroid Build Coastguard Worker   total_size_ -= size;
141*6777b538SAndroid Build Coastguard Worker }
142*6777b538SAndroid Build Coastguard Worker 
GetSizeToWrite() const143*6777b538SAndroid Build Coastguard Worker int HttpConnection::QueuedWriteIOBuffer::GetSizeToWrite() const {
144*6777b538SAndroid Build Coastguard Worker   if (IsEmpty()) {
145*6777b538SAndroid Build Coastguard Worker     DCHECK_EQ(0, total_size_);
146*6777b538SAndroid Build Coastguard Worker     return 0;
147*6777b538SAndroid Build Coastguard Worker   }
148*6777b538SAndroid Build Coastguard Worker   DCHECK_GE(data_, pending_data_.front()->data());
149*6777b538SAndroid Build Coastguard Worker   int consumed = static_cast<int>(data_ - pending_data_.front()->data());
150*6777b538SAndroid Build Coastguard Worker   DCHECK_GT(static_cast<int>(pending_data_.front()->size()), consumed);
151*6777b538SAndroid Build Coastguard Worker   return pending_data_.front()->size() - consumed;
152*6777b538SAndroid Build Coastguard Worker }
153*6777b538SAndroid Build Coastguard Worker 
HttpConnection(int id,std::unique_ptr<StreamSocket> socket)154*6777b538SAndroid Build Coastguard Worker HttpConnection::HttpConnection(int id, std::unique_ptr<StreamSocket> socket)
155*6777b538SAndroid Build Coastguard Worker     : id_(id),
156*6777b538SAndroid Build Coastguard Worker       socket_(std::move(socket)),
157*6777b538SAndroid Build Coastguard Worker       read_buf_(base::MakeRefCounted<ReadIOBuffer>()),
158*6777b538SAndroid Build Coastguard Worker       write_buf_(base::MakeRefCounted<QueuedWriteIOBuffer>()) {}
159*6777b538SAndroid Build Coastguard Worker 
160*6777b538SAndroid Build Coastguard Worker HttpConnection::~HttpConnection() = default;
161*6777b538SAndroid Build Coastguard Worker 
SetWebSocket(std::unique_ptr<WebSocket> web_socket)162*6777b538SAndroid Build Coastguard Worker void HttpConnection::SetWebSocket(std::unique_ptr<WebSocket> web_socket) {
163*6777b538SAndroid Build Coastguard Worker   DCHECK(!web_socket_);
164*6777b538SAndroid Build Coastguard Worker   web_socket_ = std::move(web_socket);
165*6777b538SAndroid Build Coastguard Worker }
166*6777b538SAndroid Build Coastguard Worker 
167*6777b538SAndroid Build Coastguard Worker }  // namespace net
168