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