]> Cypherpunks.ru repositories - gostls13.git/commitdiff
[release-branch.go1.21] net/http: regenerate h2_bundle.go
authorDamien Neil <dneil@google.com>
Fri, 6 Oct 2023 21:00:28 +0000 (14:00 -0700)
committerGopher Robot <gobot@golang.org>
Tue, 10 Oct 2023 16:10:13 +0000 (16:10 +0000)
Pull in a security fix from x/net/http2:
http2: limit maximum handler goroutines to MaxConcurrentStreamso

For #63417
Fixes #63427
Fixes CVE-2023-39325

Change-Id: I70626734e6d56edf508f27a5b055ddf96d806eeb
Reviewed-on: https://team-review.git.corp.google.com/c/golang/go-private/+/2047402
Reviewed-by: Tatiana Bradley <tatianabradley@google.com>
Run-TryBot: Damien Neil <dneil@google.com>
Reviewed-by: Ian Cottrell <iancottrell@google.com>
Reviewed-on: https://go-review.googlesource.com/c/go/+/534235
Auto-Submit: Dmitri Shuralyov <dmitshur@google.com>
Reviewed-by: Michael Pratt <mpratt@google.com>
Reviewed-by: Damien Neil <dneil@google.com>
Reviewed-by: Dmitri Shuralyov <dmitshur@google.com>
LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>

src/cmd/internal/moddeps/moddeps_test.go
src/net/http/h2_bundle.go

index ae890b66cb479cb848f0a12149df8c954f3caca5..f28e0bc077afa26b91508cdb4e914bf1d0bf5ae0 100644 (file)
@@ -33,6 +33,8 @@ import (
 // See issues 36852, 41409, and 43687.
 // (Also see golang.org/issue/27348.)
 func TestAllDependencies(t *testing.T) {
+       t.Skip("TODO(#63427): 1.21.3 contains unreleased changes from vendored modules")
+
        goBin := testenv.GoToolPath(t)
 
        // Ensure that all packages imported within GOROOT
index dc3e099c833d3731462dcc4e296a45b678346d67..9cd6a3490fcd10be36371d4b4f336bb72ae1b124 100644 (file)
@@ -4393,9 +4393,11 @@ type http2serverConn struct {
        advMaxStreams               uint32 // our SETTINGS_MAX_CONCURRENT_STREAMS advertised the client
        curClientStreams            uint32 // number of open streams initiated by the client
        curPushedStreams            uint32 // number of open streams initiated by server push
+       curHandlers                 uint32 // number of running handler goroutines
        maxClientStreamID           uint32 // max ever seen from client (odd), or 0 if there have been no client requests
        maxPushPromiseID            uint32 // ID of the last push promise (even), or 0 if there have been no pushes
        streams                     map[uint32]*http2stream
+       unstartedHandlers           []http2unstartedHandler
        initialStreamSendWindowSize int32
        maxFrameSize                int32
        peerMaxHeaderListSize       uint32            // zero means unknown (default)
@@ -4796,6 +4798,8 @@ func (sc *http2serverConn) serve() {
                                        return
                                case http2gracefulShutdownMsg:
                                        sc.startGracefulShutdownInternal()
+                               case http2handlerDoneMsg:
+                                       sc.handlerDone()
                                default:
                                        panic("unknown timer")
                                }
@@ -4843,6 +4847,7 @@ var (
        http2idleTimerMsg        = new(http2serverMessage)
        http2shutdownTimerMsg    = new(http2serverMessage)
        http2gracefulShutdownMsg = new(http2serverMessage)
+       http2handlerDoneMsg      = new(http2serverMessage)
 )
 
 func (sc *http2serverConn) onSettingsTimer() { sc.sendServeMsg(http2settingsTimerMsg) }
@@ -5842,8 +5847,7 @@ func (sc *http2serverConn) processHeaders(f *http2MetaHeadersFrame) error {
                }
        }
 
-       go sc.runHandler(rw, req, handler)
-       return nil
+       return sc.scheduleHandler(id, rw, req, handler)
 }
 
 func (sc *http2serverConn) upgradeRequest(req *Request) {
@@ -5863,6 +5867,10 @@ func (sc *http2serverConn) upgradeRequest(req *Request) {
                sc.conn.SetReadDeadline(time.Time{})
        }
 
+       // This is the first request on the connection,
+       // so start the handler directly rather than going
+       // through scheduleHandler.
+       sc.curHandlers++
        go sc.runHandler(rw, req, sc.handler.ServeHTTP)
 }
 
@@ -6103,8 +6111,62 @@ func (sc *http2serverConn) newResponseWriter(st *http2stream, req *Request) *htt
        return &http2responseWriter{rws: rws}
 }
 
+type http2unstartedHandler struct {
+       streamID uint32
+       rw       *http2responseWriter
+       req      *Request
+       handler  func(ResponseWriter, *Request)
+}
+
+// scheduleHandler starts a handler goroutine,
+// or schedules one to start as soon as an existing handler finishes.
+func (sc *http2serverConn) scheduleHandler(streamID uint32, rw *http2responseWriter, req *Request, handler func(ResponseWriter, *Request)) error {
+       sc.serveG.check()
+       maxHandlers := sc.advMaxStreams
+       if sc.curHandlers < maxHandlers {
+               sc.curHandlers++
+               go sc.runHandler(rw, req, handler)
+               return nil
+       }
+       if len(sc.unstartedHandlers) > int(4*sc.advMaxStreams) {
+               return sc.countError("too_many_early_resets", http2ConnectionError(http2ErrCodeEnhanceYourCalm))
+       }
+       sc.unstartedHandlers = append(sc.unstartedHandlers, http2unstartedHandler{
+               streamID: streamID,
+               rw:       rw,
+               req:      req,
+               handler:  handler,
+       })
+       return nil
+}
+
+func (sc *http2serverConn) handlerDone() {
+       sc.serveG.check()
+       sc.curHandlers--
+       i := 0
+       maxHandlers := sc.advMaxStreams
+       for ; i < len(sc.unstartedHandlers); i++ {
+               u := sc.unstartedHandlers[i]
+               if sc.streams[u.streamID] == nil {
+                       // This stream was reset before its goroutine had a chance to start.
+                       continue
+               }
+               if sc.curHandlers >= maxHandlers {
+                       break
+               }
+               sc.curHandlers++
+               go sc.runHandler(u.rw, u.req, u.handler)
+               sc.unstartedHandlers[i] = http2unstartedHandler{} // don't retain references
+       }
+       sc.unstartedHandlers = sc.unstartedHandlers[i:]
+       if len(sc.unstartedHandlers) == 0 {
+               sc.unstartedHandlers = nil
+       }
+}
+
 // Run on its own goroutine.
 func (sc *http2serverConn) runHandler(rw *http2responseWriter, req *Request, handler func(ResponseWriter, *Request)) {
+       defer sc.sendServeMsg(http2handlerDoneMsg)
        didPanic := true
        defer func() {
                rw.rws.stream.cancelCtx()