| // Copyright (c) 2012 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. |
| |
| #ifndef QUICHE_QUIC_TEST_TOOLS_SIMULATOR_QUIC_ENDPOINT_H_ |
| #define QUICHE_QUIC_TEST_TOOLS_SIMULATOR_QUIC_ENDPOINT_H_ |
| |
| #include "net/third_party/quiche/src/quic/core/crypto/null_decrypter.h" |
| #include "net/third_party/quiche/src/quic/core/crypto/null_encrypter.h" |
| #include "net/third_party/quiche/src/quic/core/quic_connection.h" |
| #include "net/third_party/quiche/src/quic/core/quic_default_packet_writer.h" |
| #include "net/third_party/quiche/src/quic/core/quic_packets.h" |
| #include "net/third_party/quiche/src/quic/core/quic_stream_frame_data_producer.h" |
| #include "net/third_party/quiche/src/quic/core/quic_trace_visitor.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_containers.h" |
| #include "net/third_party/quiche/src/quic/test_tools/simple_session_notifier.h" |
| #include "net/third_party/quiche/src/quic/test_tools/simulator/link.h" |
| #include "net/third_party/quiche/src/quic/test_tools/simulator/queue.h" |
| |
| namespace quic { |
| namespace simulator { |
| |
| // Size of the TX queue used by the kernel/NIC. 1000 is the Linux |
| // kernel default. |
| const QuicByteCount kTxQueueSize = 1000; |
| |
| // Generate a random local network host-port tuple based on the name of the |
| // endpoint. |
| QuicSocketAddress GetAddressFromName(std::string name); |
| |
| // A QUIC connection endpoint. Wraps around QuicConnection. In order to |
| // initiate a transfer, the caller has to call AddBytesToTransfer(). The data |
| // transferred is always the same and is always transferred on a single stream. |
| // The endpoint receives all packets addressed to it, and verifies that the data |
| // received is what it's supposed to be. |
| class QuicEndpoint : public Endpoint, |
| public UnconstrainedPortInterface, |
| public Queue::ListenerInterface, |
| public QuicConnectionVisitorInterface, |
| public SessionNotifierInterface { |
| public: |
| QuicEndpoint(Simulator* simulator, |
| std::string name, |
| std::string peer_name, |
| Perspective perspective, |
| QuicConnectionId connection_id); |
| ~QuicEndpoint() override; |
| |
| inline QuicConnection* connection() { return &connection_; } |
| QuicByteCount bytes_to_transfer() const; |
| QuicByteCount bytes_transferred() const; |
| QuicByteCount bytes_received() const; |
| inline size_t write_blocked_count() { return write_blocked_count_; } |
| inline bool wrong_data_received() const { return wrong_data_received_; } |
| |
| // Send |bytes| bytes. Initiates the transfer if one is not already in |
| // progress. |
| void AddBytesToTransfer(QuicByteCount bytes); |
| |
| // Drop the next packet upon receipt. |
| void DropNextIncomingPacket(); |
| |
| // UnconstrainedPortInterface method. Called whenever the endpoint receives a |
| // packet. |
| void AcceptPacket(std::unique_ptr<Packet> packet) override; |
| |
| // Enables logging of the connection trace at the end of the unit test. |
| void RecordTrace(); |
| |
| // Begin Endpoint implementation. |
| UnconstrainedPortInterface* GetRxPort() override; |
| void SetTxPort(ConstrainedPortInterface* port) override; |
| // End Endpoint implementation. |
| |
| // Actor method. |
| void Act() override {} |
| |
| // Queue::ListenerInterface method. |
| void OnPacketDequeued() override; |
| |
| // Begin QuicConnectionVisitorInterface implementation. |
| void OnStreamFrame(const QuicStreamFrame& frame) override; |
| void OnCryptoFrame(const QuicCryptoFrame& frame) override; |
| void OnCanWrite() override; |
| bool SendProbingData() override; |
| bool WillingAndAbleToWrite() const override; |
| bool HasPendingHandshake() const override; |
| bool ShouldKeepConnectionAlive() const override; |
| |
| void OnWindowUpdateFrame(const QuicWindowUpdateFrame& /*frame*/) override {} |
| void OnBlockedFrame(const QuicBlockedFrame& /*frame*/) override {} |
| void OnRstStream(const QuicRstStreamFrame& /*frame*/) override {} |
| void OnGoAway(const QuicGoAwayFrame& /*frame*/) override {} |
| void OnMessageReceived(QuicStringPiece /*message*/) override {} |
| void OnConnectionClosed(const QuicConnectionCloseFrame& /*frame*/, |
| ConnectionCloseSource /*source*/) override {} |
| void OnWriteBlocked() override {} |
| void OnSuccessfulVersionNegotiation( |
| const ParsedQuicVersion& /*version*/) override {} |
| void OnPacketReceived(const QuicSocketAddress& /*self_address*/, |
| const QuicSocketAddress& /*peer_address*/, |
| bool /*is_connectivity_probe*/) override {} |
| void OnCongestionWindowChange(QuicTime /*now*/) override {} |
| void OnConnectionMigration(AddressChangeType /*type*/) override {} |
| void OnPathDegrading() override {} |
| void OnAckNeedsRetransmittableFrame() override {} |
| void SendPing() override {} |
| bool AllowSelfAddressChange() const override; |
| void OnForwardProgressConfirmed() override {} |
| bool OnMaxStreamsFrame(const QuicMaxStreamsFrame& /*frame*/) override { |
| return true; |
| } |
| bool OnStreamsBlockedFrame( |
| const QuicStreamsBlockedFrame& /*frame*/) override { |
| return true; |
| } |
| bool OnStopSendingFrame(const QuicStopSendingFrame& /*frame*/) override { |
| return true; |
| } |
| |
| // End QuicConnectionVisitorInterface implementation. |
| |
| // Begin SessionNotifierInterface methods: |
| bool OnFrameAcked(const QuicFrame& frame, |
| QuicTime::Delta ack_delay_time, |
| QuicTime receive_timestamp) override; |
| void OnStreamFrameRetransmitted(const QuicStreamFrame& /*frame*/) override {} |
| void OnFrameLost(const QuicFrame& frame) override; |
| void RetransmitFrames(const QuicFrames& frames, |
| TransmissionType type) override; |
| bool IsFrameOutstanding(const QuicFrame& frame) const override; |
| bool HasUnackedCryptoData() const override; |
| bool HasUnackedStreamData() const override; |
| // End SessionNotifierInterface implementation. |
| |
| private: |
| // A Writer object that writes into the |nic_tx_queue_|. |
| class Writer : public QuicPacketWriter { |
| public: |
| explicit Writer(QuicEndpoint* endpoint); |
| ~Writer() override; |
| |
| WriteResult WritePacket(const char* buffer, |
| size_t buf_len, |
| const QuicIpAddress& self_address, |
| const QuicSocketAddress& peer_address, |
| PerPacketOptions* options) override; |
| bool IsWriteBlocked() const override; |
| void SetWritable() override; |
| QuicByteCount GetMaxPacketSize( |
| const QuicSocketAddress& peer_address) const override; |
| bool SupportsReleaseTime() const override; |
| bool IsBatchMode() const override; |
| char* GetNextWriteLocation(const QuicIpAddress& self_address, |
| const QuicSocketAddress& peer_address) override; |
| WriteResult Flush() override; |
| |
| private: |
| QuicEndpoint* endpoint_; |
| |
| bool is_blocked_; |
| }; |
| |
| // The producer outputs the repetition of the same byte. That sequence is |
| // verified by the receiver. |
| class DataProducer : public QuicStreamFrameDataProducer { |
| public: |
| WriteStreamDataResult WriteStreamData(QuicStreamId id, |
| QuicStreamOffset offset, |
| QuicByteCount data_length, |
| QuicDataWriter* writer) override; |
| bool WriteCryptoData(EncryptionLevel level, |
| QuicStreamOffset offset, |
| QuicByteCount data_length, |
| QuicDataWriter* writer) override; |
| }; |
| |
| // Write stream data until |bytes_to_transfer_| is zero or the connection is |
| // write-blocked. |
| void WriteStreamData(); |
| |
| std::string peer_name_; |
| |
| Writer writer_; |
| DataProducer producer_; |
| // The queue for the outgoing packets. In reality, this might be either on |
| // the network card, or in the kernel, but for concreteness we assume it's on |
| // the network card. |
| Queue nic_tx_queue_; |
| QuicConnection connection_; |
| |
| QuicByteCount bytes_to_transfer_; |
| QuicByteCount bytes_transferred_; |
| |
| // Counts the number of times the writer became write-blocked. |
| size_t write_blocked_count_; |
| |
| // Set to true if the endpoint receives stream data different from what it |
| // expects. |
| bool wrong_data_received_; |
| |
| // If true, drop the next packet when receiving it. |
| bool drop_next_packet_; |
| |
| // Record of received offsets in the data stream. |
| QuicIntervalSet<QuicStreamOffset> offsets_received_; |
| |
| std::unique_ptr<test::SimpleSessionNotifier> notifier_; |
| std::unique_ptr<QuicTraceVisitor> trace_visitor_; |
| }; |
| |
| // Multiplexes multiple connections at the same host on the network. |
| class QuicEndpointMultiplexer : public Endpoint, |
| public UnconstrainedPortInterface { |
| public: |
| QuicEndpointMultiplexer(std::string name, |
| const std::vector<QuicEndpoint*>& endpoints); |
| ~QuicEndpointMultiplexer() override; |
| |
| // Receives a packet and passes it to the specified endpoint if that endpoint |
| // is one of the endpoints being multiplexed, otherwise ignores the packet. |
| void AcceptPacket(std::unique_ptr<Packet> packet) override; |
| UnconstrainedPortInterface* GetRxPort() override; |
| |
| // Sets the egress port for all the endpoints being multiplexed. |
| void SetTxPort(ConstrainedPortInterface* port) override; |
| |
| void Act() override {} |
| |
| private: |
| QuicUnorderedMap<std::string, QuicEndpoint*> mapping_; |
| }; |
| |
| } // namespace simulator |
| } // namespace quic |
| |
| #endif // QUICHE_QUIC_TEST_TOOLS_SIMULATOR_QUIC_ENDPOINT_H_ |