From: Heng Hong Lee Date: Tue, 23 Aug 2016 23:31:17 +0000 (-0700) Subject: AsyncSSLSocket getRawBytes fix X-Git-Tag: v2016.08.29.00~22 X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=4d37864c657590b5460ee02eb68fd0b3b22237f7;p=folly.git AsyncSSLSocket getRawBytes fix Summary: While adding logging around our socket and looking into `AsyncSSLSocket` it seems like the data that is actually written into the socket is not correctly attributed. I added logs and printed out what happens on the socket layer, P56563098 in this paste you can see that the `[fishhook]` logs are actually those added in D3698728. Those are the actual bytes written onto the socket, in the paste, the bytes that are written by the AsyncSocket::bioWrite method are the ones that actually get written onto the socket. some of the bytes written into the bio comes from bf_buff.c and bss_mem.c which are not eventually attributed to a socket message and will be incorrectly counted when getRawBytesWritten/Read invoked on AsyncSSLSocket.cpp Unfortunately/Fortunately this issue is not symmetrical and does not manifest in the getRawBytesReceived in AsyncSSLSocket, reading the bio for read bytes correctly attributes the actual number of bytes written on the socket. moreover, pulling the asyncsocket data for getRawBytesRead doesnt give the full read bytes on wire because SSL_connect and SSL_read dont return the number of bytes they read but return the number of bytes without the TLS bytes used. siyengar seems like a right person to look at this. so adding him here. Would love to discuss more about this and am open to iterating more on this solution Reviewed By: knekritz Differential Revision: D3698744 fbshipit-source-id: 541aa478778b9607f51db194fcbfe28bd23c737f --- diff --git a/folly/io/async/AsyncSSLSocket.cpp b/folly/io/async/AsyncSSLSocket.cpp index 5efa402e..05553068 100644 --- a/folly/io/async/AsyncSSLSocket.cpp +++ b/folly/io/async/AsyncSSLSocket.cpp @@ -417,10 +417,19 @@ void AsyncSSLSocket::setEorTracking(bool track) { } size_t AsyncSSLSocket::getRawBytesWritten() const { + // The bio(s) in the write path are in a chain + // each bio flushes to the next and finally written into the socket + // to get the rawBytesWritten on the socket, + // get the write bytes of the last bio BIO *b; if (!ssl_ || !(b = SSL_get_wbio(ssl_))) { return 0; } + BIO* next = BIO_next(b); + while (next != NULL) { + b = next; + next = BIO_next(b); + } return BIO_number_written(b); }