|  | // 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/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/quic/core/crypto/crypto_handshake.h" | 
|  | #include "net/quic/core/crypto/quic_random.h" | 
|  | #include "net/quic/core/quic_crypto_stream.h" | 
|  | #include "net/quic/core/quic_data_reader.h" | 
|  | #include "net/quic/core/quic_packets.h" | 
|  | #include "net/socket/udp_server_socket.h" | 
|  | #include "net/tools/quic/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 * kMaxPacketSize; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | QuicSimpleServer::QuicSimpleServer( | 
|  | std::unique_ptr<ProofSource> proof_source, | 
|  | const QuicConfig& config, | 
|  | const QuicCryptoServerConfig::ConfigOptions& crypto_config_options, | 
|  | const QuicTransportVersionVector& supported_versions, | 
|  | QuicHttpResponseCache* response_cache) | 
|  | : version_manager_(supported_versions), | 
|  | helper_( | 
|  | new QuicChromiumConnectionHelper(&clock_, QuicRandom::GetInstance())), | 
|  | alarm_factory_(new QuicChromiumAlarmFactory( | 
|  | base::ThreadTaskRunnerHandle::Get().get(), | 
|  | &clock_)), | 
|  | config_(config), | 
|  | crypto_config_options_(crypto_config_options), | 
|  | crypto_config_(kSourceAddressTokenSecret, | 
|  | QuicRandom::GetInstance(), | 
|  | std::move(proof_source)), | 
|  | read_pending_(false), | 
|  | synchronous_read_count_(0), | 
|  | read_buffer_(new IOBufferWithSize(kReadBufferSize)), | 
|  | response_cache_(response_cache), | 
|  | 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() == | 
|  | kMinimumFlowControlSendWindow) { | 
|  | config_.SetInitialStreamFlowControlWindowToSend( | 
|  | kInitialStreamFlowControlWindow); | 
|  | } | 
|  | if (config_.GetInitialSessionFlowControlWindowToSend() == | 
|  | kMinimumFlowControlSendWindow) { | 
|  | config_.SetInitialSessionFlowControlWindowToSend( | 
|  | kInitialSessionFlowControlWindow); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<CryptoHandshakeMessage> scfg(crypto_config_.AddDefaultConfig( | 
|  | helper_->GetRandomGenerator(), helper_->GetClock(), | 
|  | crypto_config_options_)); | 
|  | } | 
|  |  | 
|  | QuicSimpleServer::~QuicSimpleServer() {} | 
|  |  | 
|  | 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>(kDefaultSocketReceiveBuffer)); | 
|  | if (rc < 0) { | 
|  | LOG(ERROR) << "SetReceiveBufferSize() failed: " << ErrorToString(rc); | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | rc = socket->SetSendBufferSize(20 * 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 QuicSimpleDispatcher( | 
|  | config_, &crypto_config_, &version_manager_, | 
|  | std::unique_ptr<QuicConnectionHelperInterface>(helper_), | 
|  | std::unique_ptr<QuicCryptoServerStream::Helper>( | 
|  | new QuicSimpleServerSessionHelper(QuicRandom::GetInstance())), | 
|  | std::unique_ptr<QuicAlarmFactory>(alarm_factory_), response_cache_)); | 
|  | 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(); | 
|  |  | 
|  | 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; | 
|  | } | 
|  |  | 
|  | QuicReceivedPacket packet(read_buffer_->data(), result, | 
|  | helper_->GetClock()->Now(), false); | 
|  | dispatcher_->ProcessPacket( | 
|  | QuicSocketAddress(QuicSocketAddressImpl(server_address_)), | 
|  | QuicSocketAddress(QuicSocketAddressImpl(client_address_)), packet); | 
|  |  | 
|  | StartReading(); | 
|  | } | 
|  |  | 
|  | }  // namespace net |