Add Pigeon USPS GSO Support via quic::PigeonWriter.
This CL adds quic::PigeonWriter and quic::PigeonSocket classes, with extensive unit tests. It refactors the old gfe2::RawSocket into a quic::RawSocketInterface and gfe2::RawQuicWriter into quic::RawPacketWriterBase. All gfe/gfe2/quic classes continue to work in the same way as before, only refactored. The changes in the CL are transparent to gfe/gfe2/quic.
The new PigeonSocket uses a BufferAllocator, which is a freelist of shared memory buffers of size 2048 bytes. The socket can externally share shared memory buffers via PigeonSocket::AcquireSegmentBufferFromSharedMemory(). The PigeonWriter uses this to support "GetNextWriteLocation()" for the QUIC connection to serialize packets directly into shared memory. This eliminates a memcpy() in the critical path.
The PigeonWriter and PigeonSocket classes support GSO by batching up to --quic_pigeon_socket_max_gso_segments number of segments per write. The PigeonWriter can also be used in pass-through mode by disabling GSO via flag --quic_pigeon_writer_enable_udp_gso.
If flag --ustreamer_quic_use_new_pigeon_writer is true, and the egress method is USPS, then Ustreamer will create the new quic::PigeonWriter rather than gfe2::RawQuicWriter.
gfe-relnote: GFE related changes refactors gfe's PigeonSocketWriteBlockHandler to net/quic/pigeon_socket.h
PiperOrigin-RevId: 304279221
Change-Id: I919f8a7b8f2f039f07c078f11f2413dd6f39e12a
diff --git a/quic/core/quic_packet_writer.h b/quic/core/quic_packet_writer.h
index 5f6b681..bd10523 100644
--- a/quic/core/quic_packet_writer.h
+++ b/quic/core/quic_packet_writer.h
@@ -78,6 +78,26 @@
//
// Options must be either null, or created for the particular QuicPacketWriter
// implementation. Options may be ignored, depending on the implementation.
+ //
+ // Some comment about memory management if |buffer| was previously acquired
+ // by a call to "GetNextWriteLocation()":
+ //
+ // a) When WRITE_STATUS_OK is returned, the caller expects the writer owns the
+ // packet buffers and they will be released when the write finishes.
+ //
+ // b) When this function returns any status >= WRITE_STATUS_ERROR, the caller
+ // expects the writer releases the buffer (if needed) before the function
+ // returns.
+ //
+ // c) When WRITE_STATUS_BLOCKED is returned, the caller makes a copy of the
+ // buffer and will retry after unblock, so if |payload| is allocated from
+ // GetNextWriteLocation(), it
+ // 1) needs to be released before return, and
+ // 2) the content of |payload| should not change after return.
+ //
+ // d) When WRITE_STATUS_BLOCKED_DATA_BUFFERED is returned, the caller expects
+ // 1) the writer owns the packet buffers, and 2) the writer will re-send the
+ // packet when it unblocks.
virtual WriteResult WritePacket(const char* buffer,
size_t buf_len,
const QuicIpAddress& self_address,
@@ -121,9 +141,13 @@
// Try send all buffered packets.
// - Return WriteResult(WRITE_STATUS_OK, <bytes_flushed>) if all buffered
// packets were sent successfully.
- // - Return WRITE_STATUS_BLOCKED, or an error status, if the underlying socket
- // is blocked or returned an error while sending. Some packets may have been
- // sent, packets not sent will stay in the internal buffer.
+ // - Return WRITE_STATUS_BLOCKED if the underlying socket is blocked while
+ // sending. Some packets may have been sent, packets not sent will stay in
+ // the internal buffer.
+ // - Return a status >= WRITE_STATUS_ERROR if an error was encuontered while
+ // sending. As this is not a re-tryable error, any batched packets which
+ // were on memory acquired via GetNextWriteLocation() should be released and
+ // the batch should be dropped.
virtual WriteResult Flush() = 0;
};