Tune http2 window and frame sizes

http2 maintains its own frame ACK window per stream.  While the defaults
work well in most cases, for large binary uploads, like Redfish
UpdateService, the relatively small default window size of 16KB leads to
slower performance than http1.  While it's not expected to see a
performance improvement, we would prefer to not see a regression for a
normal use case.

Update the HTTP2 max frame size to 16KB.  Setting the internal buffer to
the same size + the http2 header allows clocking in the entire frame in
one async read.  Note, setting the value higher than 16KB doesn't appear
to allow curl to send larger frames.

Also update the HTTP window size to 512KB, or 32 times the max frame
size. Note, all streams including the control stream are set to this
value, which, while somewhat arbitrary, allows for continued
UpdateService pushing without pauses for window ACK.

Tested:
POST /redfish/v1/UpdateService/update-multipart
Of an arbitrary 100MB file through curl shows that --http1.1 option and
--http2 option are within 5% of the same upload time.

Change-Id: I7ff6296a9cc0794aad63f5058620c0f1fb9299e3
Signed-off-by: Ed Tanous <etanous@nvidia.com>
diff --git a/http/http2_connection.hpp b/http/http2_connection.hpp
index df41ff6..7b90ff3 100644
--- a/http/http2_connection.hpp
+++ b/http/http2_connection.hpp
@@ -113,9 +113,23 @@
         BMCWEB_LOG_DEBUG("send_server_connection_header()");
 
         uint32_t maxStreams = 4;
-        std::array<nghttp2_settings_entry, 2> iv = {
-            {{NGHTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, maxStreams},
-             {NGHTTP2_SETTINGS_ENABLE_PUSH, 0}}};
+
+        // Both of these settings were found experimentally to allow a single
+        // fast stream to upload at a rate equivalent to http1.1  They will
+        // likely be tuned in the future.
+        uint32_t maxFrameSize = 1 << 14;
+        uint32_t windowSize = 1 << 20;
+        std::array<nghttp2_settings_entry, 4> iv = {{
+            {NGHTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, maxStreams},
+            {NGHTTP2_SETTINGS_ENABLE_PUSH, 0},
+            // Set an approximately 1MB window size
+            {NGHTTP2_SETTINGS_INITIAL_WINDOW_SIZE, windowSize},
+            {NGHTTP2_SETTINGS_MAX_FRAME_SIZE, maxFrameSize},
+        }};
+        if (ngSession.setLocalWindowSize(NGHTTP2_FLAG_NONE, 0, 1 << 20) != 0)
+        {
+            BMCWEB_LOG_ERROR("Failed to set local window size");
+        }
         int rv = ngSession.submitSettings(iv);
         if (rv != 0)
         {
@@ -365,7 +379,7 @@
         nghttp2_session* /* session */, uint8_t flags, int32_t streamId,
         const uint8_t* data, size_t len, void* userData)
     {
-        BMCWEB_LOG_DEBUG("on_frame_recv_callback");
+        BMCWEB_LOG_DEBUG("onDataChunkRecvStatic");
         if (userData == nullptr)
         {
             BMCWEB_LOG_CRITICAL("user data was null?");
@@ -398,7 +412,8 @@
                                          const nghttp2_frame* frame,
                                          void* userData)
     {
-        BMCWEB_LOG_DEBUG("on_frame_recv_callback");
+        BMCWEB_LOG_DEBUG("on_frame_recv_callback.  Frame type {}",
+                         static_cast<int>(frame->hd.type));
         if (userData == nullptr)
         {
             BMCWEB_LOG_CRITICAL("user data was null?");
@@ -533,6 +548,11 @@
             BMCWEB_LOG_DEBUG("create stream for id {}", frame.hd.stream_id);
 
             streams[frame.hd.stream_id];
+            if (ngSession.setLocalWindowSize(
+                    NGHTTP2_FLAG_NONE, frame.hd.stream_id, 16384 * 32) != 0)
+            {
+                BMCWEB_LOG_ERROR("Failed to set local window size");
+            }
         }
         return 0;
     }
diff --git a/http/nghttp2_adapters.hpp b/http/nghttp2_adapters.hpp
index ff5e708..15bde7c 100644
--- a/http/nghttp2_adapters.hpp
+++ b/http/nghttp2_adapters.hpp
@@ -173,6 +173,12 @@
                                        headers.size(), dataPrd);
     }
 
+    int setLocalWindowSize(uint8_t flags, int32_t stream_id, int32_t windowSize)
+    {
+        return nghttp2_session_set_local_window_size(ptr, flags, stream_id,
+                                                     windowSize);
+    }
+
   private:
     nghttp2_session* ptr = nullptr;
 };
diff --git a/test/http/http2_connection_test.cpp b/test/http/http2_connection_test.cpp
index 0fb0de6..a6bc81c 100644
--- a/test/http/http2_connection_test.cpp
+++ b/test/http/http2_connection_test.cpp
@@ -133,16 +133,27 @@
         &handler, date, HttpType::HTTP, nullptr);
     conn->start();
 
-    std::array<std::string_view, 5> expectedPrefix = {
-        // Settings frame size 13
-        "\x00\x00\x0c\x04\x00\x00\x00\x00\x00"sv,
+    std::array<std::string_view, 9> expectedPrefix = {
+        // Settings frame size 24
+        "\x00\x00\x18\x04\x00\x00\x00\x00\x00"sv,
         // 4 max concurrent streams
         "\x00\x03\x00\x00\x00\x04"sv,
         // Enable push = false
         "\x00\x02\x00\x00\x00\x00"sv,
+        // Max window size 1 << 20
+        "\x00\x04\x00\x10\x00\x00"sv,
+        // Max frame size 1 << 14
+        "\x00\x05\x00\x00\x40\x00"sv,
+
+        // Frame window update stream 0
+        "\x00\x00\x04\x08\x00\x00\x00\x00\x00\x00\x0f\x00\x01"sv,
+
         // Settings ACK from server to client
         "\x00\x00\x00\x04\x01\x00\x00\x00\x00"sv,
 
+        // Window update stream 1
+        "\x00\x00\x04\x08\x00\x00\x00\x00\x01\x00\x07\x00\x01"sv,
+
         // Start Headers frame stream 1, size 0x005f
         "\x00\x00\x5f\x01\x04\x00\x00\x00\x01"sv,
     };
@@ -161,6 +172,7 @@
     {
         expectedPrefixSize += prefix.size();
     }
+
     // Run until we receive the expected amount of data
     while (outStr.size() <
            expectedPrefixSize + headerSize + expectedPostfix.size())