From e969dc64018a6722118496bc1c2fc76c141bacba Mon Sep 17 00:00:00 2001 From: Aman Sharma Date: Wed, 22 Jan 2025 14:09:17 -0800 Subject: [PATCH] Make a writemGSO API with just iovecs [2/n] Summary: **Background for the diff stack**: We'd like to excise the usage of folly from a library that we maintain (quic), which is why we'd like to have a version of `writemGSO` that takes in iovecs instead of IOBufs. **What this diff is doing**: I'd like to use `fillMsgVec` within the version of `writemGSO` that takes in iovecs instead of IOBufs. I'm changing the signature of `fillMsgVec` so that it doesn't need to take in any IOBufs. Instead, it gets the lengths of the buffers from the passed in array `messageIovLens`, which is filled in by `fillIoVec`. Reviewed By: jbeshay Differential Revision: D67873321 fbshipit-source-id: 93181cc69f51bce84e985c17b5f0d660e12c620e --- folly/io/async/AsyncUDPSocket.cpp | 31 ++++++++++++++++++++++++------- folly/io/async/AsyncUDPSocket.h | 3 ++- 2 files changed, 26 insertions(+), 8 deletions(-) diff --git a/folly/io/async/AsyncUDPSocket.cpp b/folly/io/async/AsyncUDPSocket.cpp index e0c12791b57..86cb3b2f613 100644 --- a/folly/io/async/AsyncUDPSocket.cpp +++ b/folly/io/async/AsyncUDPSocket.cpp @@ -868,11 +868,13 @@ int AsyncUDPSocket::writemGSO( void AsyncUDPSocket::fillIoVec( const std::unique_ptr* bufs, struct iovec* iov, + size_t* messageIovLens, size_t count, size_t iov_count) { size_t remaining = iov_count; size_t iov_pos = 0; for (size_t i = 0; i < count; i++) { + messageIovLens[i] = bufs[i]->countChainElements(); auto ret = bufs[i]->fillIov(&iov[iov_pos], remaining); size_t iovec_len = ret.numIovecs; remaining -= iovec_len; @@ -882,7 +884,7 @@ void AsyncUDPSocket::fillIoVec( void AsyncUDPSocket::fillMsgVec( Range addrs, - const std::unique_ptr* bufs, + size_t* messageIovLens, size_t count, struct mmsghdr* msgvec, struct iovec* iov, @@ -905,7 +907,7 @@ void AsyncUDPSocket::fillMsgVec( msg.msg_namelen = addrs[addr_count - 1].len; } msg.msg_iov = &iov[iov_pos]; - msg.msg_iovlen = bufs[i]->countChainElements(); + msg.msg_iovlen = messageIovLens[i]; #ifdef FOLLY_HAVE_MSG_ERRQUEUE size_t controlBufSize = 1 + cmsgs_->size() * @@ -982,7 +984,7 @@ void AsyncUDPSocket::fillMsgVec( msgvec[i].msg_len = 0; - iov_pos += bufs[i]->countChainElements(); + iov_pos += messageIovLens[i]; } } @@ -1015,15 +1017,30 @@ int AsyncUDPSocket::writeImpl( FOLLY_PUSH_WARNING FOLLY_GNU_DISABLE_WARNING("-Wvla") iovec iov[BOOST_PP_IF(FOLLY_HAVE_VLA_01, iov_count, kSmallSizeMax)]; + size_t messageIovLens[BOOST_PP_IF(FOLLY_HAVE_VLA_01, count, kSmallSizeMax)]; FOLLY_POP_WARNING - fillIoVec(bufs, iov, count, iov_count); - fillMsgVec(range(addrStorage), bufs, count, msgvec, iov, options, control); + fillIoVec(bufs, iov, messageIovLens, count, iov_count); + fillMsgVec( + range(addrStorage), + messageIovLens, + count, + msgvec, + iov, + options, + control); ret = sendmmsg(fd_, msgvec, count, 0); } else { std::unique_ptr iov(new iovec[iov_count]); - fillIoVec(bufs, iov.get(), count, iov_count); + std::unique_ptr messageIovLens(new size_t[count]); + fillIoVec(bufs, iov.get(), messageIovLens.get(), count, iov_count); fillMsgVec( - range(addrStorage), bufs, count, msgvec, iov.get(), options, control); + range(addrStorage), + messageIovLens.get(), + count, + msgvec, + iov.get(), + options, + control); ret = sendmmsg(fd_, msgvec, count, 0); } diff --git a/folly/io/async/AsyncUDPSocket.h b/folly/io/async/AsyncUDPSocket.h index c980f17726b..274c174361e 100644 --- a/folly/io/async/AsyncUDPSocket.h +++ b/folly/io/async/AsyncUDPSocket.h @@ -544,12 +544,13 @@ class AsyncUDPSocket : public EventHandler { void fillIoVec( const std::unique_ptr* bufs, struct iovec* iov, + size_t* messageIovLens, size_t count, size_t iov_count); void fillMsgVec( Range addrs, - const std::unique_ptr* bufs, + size_t* messageIovLens, size_t count, struct mmsghdr* msgvec, struct iovec* iov,