blob: 6efdb6005d887d269a19f3e4d1ea38c23de3d7d4 [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// Copyright 2013 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#ifndef QUICHE_QUIC_CORE_QUIC_PACKET_WRITER_H_
6#define QUICHE_QUIC_CORE_QUIC_PACKET_WRITER_H_
7
8#include <cstddef>
bnc463f2352019-10-10 04:49:34 -07009#include <utility>
QUICHE teama6ef0a62019-03-07 20:34:33 -050010
11#include "net/third_party/quiche/src/quic/core/quic_packets.h"
12#include "net/third_party/quiche/src/quic/platform/api/quic_export.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050013#include "net/third_party/quiche/src/quic/platform/api/quic_socket_address.h"
14
15namespace quic {
16
17struct WriteResult;
18
19struct QUIC_EXPORT_PRIVATE PerPacketOptions {
20 virtual ~PerPacketOptions() {}
21
22 // Returns a heap-allocated copy of |this|.
23 //
24 // The subclass implementation of this method should look like this:
bnc463f2352019-10-10 04:49:34 -070025 // return std::make_unique<MyAwesomePerPacketOptions>(*this);
QUICHE teama6ef0a62019-03-07 20:34:33 -050026 //
27 // This method is declared pure virtual in order to ensure the subclasses
28 // would not forget to override it.
29 virtual std::unique_ptr<PerPacketOptions> Clone() const = 0;
30
wubed245d22020-05-07 10:46:00 -070031 // Specifies ideal release time delay for this packet.
QUICHE teama6ef0a62019-03-07 20:34:33 -050032 QuicTime::Delta release_time_delay = QuicTime::Delta::Zero();
wubed245d22020-05-07 10:46:00 -070033 // Whether it is allowed to send this packet without |release_time_delay|.
34 bool allow_burst = false;
QUICHE teama6ef0a62019-03-07 20:34:33 -050035};
36
37// An interface between writers and the entity managing the
38// socket (in our case the QuicDispatcher). This allows the Dispatcher to
39// control writes, and manage any writers who end up write blocked.
40// A concrete writer works in one of the two modes:
41// - PassThrough mode. This is the default mode. Caller calls WritePacket with
42// caller-allocated packet buffer. Unless the writer is blocked, each call to
43// WritePacket triggers a write using the underlying socket API.
44//
45// - Batch mode. In this mode, a call to WritePacket may not cause a packet to
46// be sent using the underlying socket API. Instead, multiple packets are
47// saved in the writer's internal buffer until they are flushed. The flush can
48// be explicit, by calling Flush, or implicit, e.g. by calling
49// WritePacket when the internal buffer is near full.
50//
51// Buffer management:
52// In Batch mode, a writer manages an internal buffer, which is large enough to
53// hold multiple packets' data. If the caller calls WritePacket with a
54// caller-allocated packet buffer, the writer will memcpy the buffer into the
55// internal buffer. Caller can also avoid this memcpy by:
56// 1. Call GetNextWriteLocation to get a pointer P into the internal buffer.
57// 2. Serialize the packet directly to P.
58// 3. Call WritePacket with P as the |buffer|.
59class QUIC_EXPORT_PRIVATE QuicPacketWriter {
60 public:
61 virtual ~QuicPacketWriter() {}
62
63 // PassThrough mode:
64 // Sends the packet out to the peer, with some optional per-packet options.
65 // If the write succeeded, the result's status is WRITE_STATUS_OK and
66 // bytes_written is populated. If the write failed, the result's status is
67 // WRITE_STATUS_BLOCKED or WRITE_STATUS_ERROR and error_code is populated.
68 //
69 // Batch mode:
70 // If the writer is blocked, return WRITE_STATUS_BLOCKED immediately.
71 // If the packet can be batched with other buffered packets, save the packet
72 // to the internal buffer.
73 // If the packet can not be batched, or the internal buffer is near full after
74 // it is buffered, the internal buffer is flushed to free up space.
75 // Return WriteResult(WRITE_STATUS_OK, <bytes_flushed>) on success. When
76 // <bytes_flushed> is zero, it means the packet is buffered and not flushed.
77 // Return WRITE_STATUS_BLOCKED if the packet is not buffered and the socket is
78 // blocked while flushing.
79 // Otherwise return an error status.
80 //
81 // Options must be either null, or created for the particular QuicPacketWriter
82 // implementation. Options may be ignored, depending on the implementation.
QUICHE teamf018b732020-04-01 15:53:51 -070083 //
84 // Some comment about memory management if |buffer| was previously acquired
85 // by a call to "GetNextWriteLocation()":
86 //
87 // a) When WRITE_STATUS_OK is returned, the caller expects the writer owns the
88 // packet buffers and they will be released when the write finishes.
89 //
90 // b) When this function returns any status >= WRITE_STATUS_ERROR, the caller
91 // expects the writer releases the buffer (if needed) before the function
92 // returns.
93 //
94 // c) When WRITE_STATUS_BLOCKED is returned, the caller makes a copy of the
95 // buffer and will retry after unblock, so if |payload| is allocated from
96 // GetNextWriteLocation(), it
97 // 1) needs to be released before return, and
98 // 2) the content of |payload| should not change after return.
99 //
100 // d) When WRITE_STATUS_BLOCKED_DATA_BUFFERED is returned, the caller expects
101 // 1) the writer owns the packet buffers, and 2) the writer will re-send the
102 // packet when it unblocks.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500103 virtual WriteResult WritePacket(const char* buffer,
104 size_t buf_len,
105 const QuicIpAddress& self_address,
106 const QuicSocketAddress& peer_address,
107 PerPacketOptions* options) = 0;
108
109 // Returns true if the network socket is not writable.
110 virtual bool IsWriteBlocked() const = 0;
111
112 // Records that the socket has become writable, for example when an EPOLLOUT
113 // is received or an asynchronous write completes.
114 virtual void SetWritable() = 0;
115
116 // Returns the maximum size of the packet which can be written using this
117 // writer for the supplied peer address. This size may actually exceed the
118 // size of a valid QUIC packet.
119 virtual QuicByteCount GetMaxPacketSize(
120 const QuicSocketAddress& peer_address) const = 0;
121
122 // Returns true if the socket supports release timestamp.
123 virtual bool SupportsReleaseTime() const = 0;
124
125 // True=Batch mode. False=PassThrough mode.
126 virtual bool IsBatchMode() const = 0;
127
wub50d4c712020-05-19 15:48:28 -0700128 // PassThrough mode: Return {nullptr, nullptr}
QUICHE teama6ef0a62019-03-07 20:34:33 -0500129 //
130 // Batch mode:
wub50d4c712020-05-19 15:48:28 -0700131 // Return the QuicPacketBuffer for the next packet. A minimum of
dschinazi66dea072019-04-09 11:41:06 -0700132 // kMaxOutgoingPacketSize is guaranteed to be available from the returned
wub50d4c712020-05-19 15:48:28 -0700133 // address. If the internal buffer does not have enough space,
134 // {nullptr, nullptr} is returned. All arguments should be identical to the
135 // follow-up call to |WritePacket|, they are here to allow advanced packet
136 // memory management in packet writers, e.g. one packet buffer pool per
137 // |peer_address|.
138 //
139 // If QuicPacketBuffer.release_buffer is !nullptr, it should be called iff
140 // the caller does not call WritePacket for the returned buffer.
141 virtual QuicPacketBuffer GetNextWriteLocation(
142 const QuicIpAddress& self_address,
143 const QuicSocketAddress& peer_address) = 0;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500144
145 // PassThrough mode: Return WriteResult(WRITE_STATUS_OK, 0).
146 //
147 // Batch mode:
148 // Try send all buffered packets.
149 // - Return WriteResult(WRITE_STATUS_OK, <bytes_flushed>) if all buffered
150 // packets were sent successfully.
QUICHE teamf018b732020-04-01 15:53:51 -0700151 // - Return WRITE_STATUS_BLOCKED if the underlying socket is blocked while
152 // sending. Some packets may have been sent, packets not sent will stay in
153 // the internal buffer.
154 // - Return a status >= WRITE_STATUS_ERROR if an error was encuontered while
155 // sending. As this is not a re-tryable error, any batched packets which
156 // were on memory acquired via GetNextWriteLocation() should be released and
157 // the batch should be dropped.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500158 virtual WriteResult Flush() = 0;
159};
160
161} // namespace quic
162
163#endif // QUICHE_QUIC_CORE_QUIC_PACKET_WRITER_H_