| // Copyright 2014 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/tools/quic/quic_simple_server.h" |
| |
| #include <string.h> |
| |
| #include "base/bind.h" |
| #include "base/location.h" |
| #include "base/single_thread_task_runner.h" |
| #include "base/threading/thread_task_runner_handle.h" |
| #include "net/base/ip_endpoint.h" |
| #include "net/base/net_errors.h" |
| #include "net/log/net_log_source.h" |
| #include "net/socket/udp_server_socket.h" |
| #include "net/third_party/quic/core/crypto/crypto_handshake.h" |
| #include "net/third_party/quic/core/crypto/quic_random.h" |
| #include "net/third_party/quic/core/quic_crypto_stream.h" |
| #include "net/third_party/quic/core/quic_data_reader.h" |
| #include "net/third_party/quic/core/quic_packets.h" |
| #include "net/third_party/quic/core/tls_server_handshaker.h" |
| #include "net/third_party/quic/tools/quic_simple_dispatcher.h" |
| #include "net/tools/quic/quic_simple_per_connection_packet_writer.h" |
| #include "net/tools/quic/quic_simple_server_packet_writer.h" |
| #include "net/tools/quic/quic_simple_server_session_helper.h" |
| |
| namespace net { |
| |
| namespace { |
| |
| const char kSourceAddressTokenSecret[] = "secret"; |
| const size_t kNumSessionsToCreatePerSocketEvent = 16; |
| |
| // Allocate some extra space so we can send an error if the client goes over |
| // the limit. |
| const int kReadBufferSize = 2 * quic::kMaxPacketSize; |
| |
| } // namespace |
| |
| QuicSimpleServer::QuicSimpleServer( |
| std::unique_ptr<quic::ProofSource> proof_source, |
| const quic::QuicConfig& config, |
| const quic::QuicCryptoServerConfig::ConfigOptions& crypto_config_options, |
| const quic::ParsedQuicVersionVector& supported_versions, |
| quic::QuicSimpleServerBackend* quic_simple_server_backend) |
| : version_manager_(supported_versions), |
| helper_( |
| new QuicChromiumConnectionHelper(&clock_, |
| quic::QuicRandom::GetInstance())), |
| alarm_factory_(new QuicChromiumAlarmFactory( |
| base::ThreadTaskRunnerHandle::Get().get(), |
| &clock_)), |
| config_(config), |
| crypto_config_options_(crypto_config_options), |
| crypto_config_(kSourceAddressTokenSecret, |
| quic::QuicRandom::GetInstance(), |
| std::move(proof_source), |
| quic::KeyExchangeSource::Default(), |
| quic::TlsServerHandshaker::CreateSslCtx()), |
| read_pending_(false), |
| synchronous_read_count_(0), |
| read_buffer_(base::MakeRefCounted<IOBufferWithSize>(kReadBufferSize)), |
| quic_simple_server_backend_(quic_simple_server_backend), |
| weak_factory_(this) { |
| Initialize(); |
| } |
| |
| void QuicSimpleServer::Initialize() { |
| #if MMSG_MORE |
| use_recvmmsg_ = true; |
| #endif |
| |
| // If an initial flow control window has not explicitly been set, then use a |
| // sensible value for a server: 1 MB for session, 64 KB for each stream. |
| const uint32_t kInitialSessionFlowControlWindow = 1 * 1024 * 1024; // 1 MB |
| const uint32_t kInitialStreamFlowControlWindow = 64 * 1024; // 64 KB |
| if (config_.GetInitialStreamFlowControlWindowToSend() == |
| quic::kMinimumFlowControlSendWindow) { |
| config_.SetInitialStreamFlowControlWindowToSend( |
| kInitialStreamFlowControlWindow); |
| } |
| if (config_.GetInitialSessionFlowControlWindowToSend() == |
| quic::kMinimumFlowControlSendWindow) { |
| config_.SetInitialSessionFlowControlWindowToSend( |
| kInitialSessionFlowControlWindow); |
| } |
| |
| std::unique_ptr<quic::CryptoHandshakeMessage> scfg( |
| crypto_config_.AddDefaultConfig(helper_->GetRandomGenerator(), |
| helper_->GetClock(), |
| crypto_config_options_)); |
| } |
| |
| QuicSimpleServer::~QuicSimpleServer() = default; |
| |
| int QuicSimpleServer::Listen(const IPEndPoint& address) { |
| std::unique_ptr<UDPServerSocket> socket( |
| new UDPServerSocket(&net_log_, NetLogSource())); |
| |
| socket->AllowAddressReuse(); |
| |
| int rc = socket->Listen(address); |
| if (rc < 0) { |
| LOG(ERROR) << "Listen() failed: " << ErrorToString(rc); |
| return rc; |
| } |
| |
| // These send and receive buffer sizes are sized for a single connection, |
| // because the default usage of QuicSimpleServer is as a test server with |
| // one or two clients. Adjust higher for use with many clients. |
| rc = socket->SetReceiveBufferSize( |
| static_cast<int32_t>(quic::kDefaultSocketReceiveBuffer)); |
| if (rc < 0) { |
| LOG(ERROR) << "SetReceiveBufferSize() failed: " << ErrorToString(rc); |
| return rc; |
| } |
| |
| rc = socket->SetSendBufferSize(20 * quic::kMaxPacketSize); |
| if (rc < 0) { |
| LOG(ERROR) << "SetSendBufferSize() failed: " << ErrorToString(rc); |
| return rc; |
| } |
| |
| rc = socket->GetLocalAddress(&server_address_); |
| if (rc < 0) { |
| LOG(ERROR) << "GetLocalAddress() failed: " << ErrorToString(rc); |
| return rc; |
| } |
| |
| DVLOG(1) << "Listening on " << server_address_.ToString(); |
| |
| socket_.swap(socket); |
| |
| dispatcher_.reset(new quic::QuicSimpleDispatcher( |
| config_, &crypto_config_, &version_manager_, |
| std::unique_ptr<quic::QuicConnectionHelperInterface>(helper_), |
| std::unique_ptr<quic::QuicCryptoServerStream::Helper>( |
| new QuicSimpleServerSessionHelper(quic::QuicRandom::GetInstance())), |
| std::unique_ptr<quic::QuicAlarmFactory>(alarm_factory_), |
| quic_simple_server_backend_)); |
| QuicSimpleServerPacketWriter* writer = |
| new QuicSimpleServerPacketWriter(socket_.get(), dispatcher_.get()); |
| dispatcher_->InitializeWithWriter(writer); |
| |
| StartReading(); |
| |
| return OK; |
| } |
| |
| void QuicSimpleServer::Shutdown() { |
| // Before we shut down the epoll server, give all active sessions a chance to |
| // notify clients that they're closing. |
| dispatcher_->Shutdown(); |
| |
| if (!socket_) { |
| return; |
| } |
| socket_->Close(); |
| socket_.reset(); |
| } |
| |
| void QuicSimpleServer::StartReading() { |
| if (synchronous_read_count_ == 0) { |
| // Only process buffered packets once per message loop. |
| dispatcher_->ProcessBufferedChlos(kNumSessionsToCreatePerSocketEvent); |
| } |
| |
| if (read_pending_) { |
| return; |
| } |
| read_pending_ = true; |
| |
| int result = socket_->RecvFrom( |
| read_buffer_.get(), read_buffer_->size(), &client_address_, |
| base::Bind(&QuicSimpleServer::OnReadComplete, base::Unretained(this))); |
| |
| if (result == ERR_IO_PENDING) { |
| synchronous_read_count_ = 0; |
| if (dispatcher_->HasChlosBuffered()) { |
| // No more packets to read, so yield before processing buffered packets. |
| base::ThreadTaskRunnerHandle::Get()->PostTask( |
| FROM_HERE, base::Bind(&QuicSimpleServer::StartReading, |
| weak_factory_.GetWeakPtr())); |
| } |
| return; |
| } |
| |
| if (++synchronous_read_count_ > 32) { |
| synchronous_read_count_ = 0; |
| // Schedule the processing through the message loop to 1) prevent infinite |
| // recursion and 2) avoid blocking the thread for too long. |
| base::ThreadTaskRunnerHandle::Get()->PostTask( |
| FROM_HERE, base::Bind(&QuicSimpleServer::OnReadComplete, |
| weak_factory_.GetWeakPtr(), result)); |
| } else { |
| OnReadComplete(result); |
| } |
| } |
| |
| void QuicSimpleServer::OnReadComplete(int result) { |
| read_pending_ = false; |
| if (result == 0) |
| result = ERR_CONNECTION_CLOSED; |
| |
| if (result < 0) { |
| LOG(ERROR) << "QuicSimpleServer read failed: " << ErrorToString(result); |
| Shutdown(); |
| return; |
| } |
| |
| quic::QuicReceivedPacket packet(read_buffer_->data(), result, |
| helper_->GetClock()->Now(), false); |
| dispatcher_->ProcessPacket( |
| quic::QuicSocketAddress(quic::QuicSocketAddressImpl(server_address_)), |
| quic::QuicSocketAddress(quic::QuicSocketAddressImpl(client_address_)), |
| packet); |
| |
| StartReading(); |
| } |
| |
| } // namespace net |