}
 
 type NodeEscState struct {
-       Curfn        *Node
-       Escflowsrc   *NodeList // flow(this, src)
-       Escretval    *NodeList // on OCALLxxx, list of dummy return values
-       Escloopdepth int32     // -1: global, 0: return variables, 1:function top level, increased inside function for every loop or label to mark scopes
-       Esclevel     Level
-       Walkgen      uint32
+       Curfn             *Node
+       Escflowsrc        *NodeList // flow(this, src)
+       Escretval         *NodeList // on OCALLxxx, list of dummy return values
+       Escloopdepth      int32     // -1: global, 0: return variables, 1:function top level, increased inside function for every loop or label to mark scopes
+       Esclevel          Level
+       Walkgen           uint32
+       Maxextraloopdepth int32
 }
 
 func (e *EscState) nodeEscState(n *Node) *NodeEscState {
                src.Op == ONAME && src.Class == PPARAM && src.Name.Curfn == dst.Name.Curfn
 }
 
+const NOTALOOPDEPTH = -1
+
 func escwalk(e *EscState, level Level, dst *Node, src *Node) {
+       escwalkBody(e, level, dst, src, NOTALOOPDEPTH)
+}
+
+func escwalkBody(e *EscState, level Level, dst *Node, src *Node, extraloopdepth int32) {
        if src.Op == OLITERAL {
                return
        }
                // convergence.
                level = level.min(srcE.Esclevel)
                if level == srcE.Esclevel {
-                       return
+                       // Have we been here already with an extraloopdepth,
+                       // or is the extraloopdepth provided no improvement on
+                       // what's already been seen?
+                       if srcE.Maxextraloopdepth >= extraloopdepth || srcE.Escloopdepth >= extraloopdepth {
+                               return
+                       }
+                       srcE.Maxextraloopdepth = extraloopdepth
                }
+       } else { // srcE.Walkgen < e.walkgen -- first time, reset this.
+               srcE.Maxextraloopdepth = NOTALOOPDEPTH
        }
 
        srcE.Walkgen = e.walkgen
        srcE.Esclevel = level
+       modSrcLoopdepth := srcE.Escloopdepth
+
+       if extraloopdepth > modSrcLoopdepth {
+               modSrcLoopdepth = extraloopdepth
+       }
 
        if Debug['m'] > 1 {
-               fmt.Printf("escwalk: level:%d depth:%d %.*s op=%v %v(%v) scope:%v[%d]\n",
-                       level, e.pdepth, e.pdepth, "\t\t\t\t\t\t\t\t\t\t", Oconv(int(src.Op), 0), Nconv(src, obj.FmtShort), Jconv(src, obj.FmtShort), e.curfnSym(src), srcE.Escloopdepth)
+               fmt.Printf("escwalk: level:%d depth:%d %.*s op=%v %v(%v) scope:%v[%d] extraloopdepth=%v\n",
+                       level, e.pdepth, e.pdepth, "\t\t\t\t\t\t\t\t\t\t", Oconv(int(src.Op), 0), Nconv(src, obj.FmtShort), Jconv(src, obj.FmtShort), e.curfnSym(src), srcE.Escloopdepth, extraloopdepth)
        }
 
        e.pdepth++
                }
        }
 
-       leaks = level.int() <= 0 && level.guaranteedDereference() <= 0 && dstE.Escloopdepth < srcE.Escloopdepth
+       leaks = level.int() <= 0 && level.guaranteedDereference() <= 0 && dstE.Escloopdepth < modSrcLoopdepth
 
        switch src.Op {
        case ONAME:
                                                Warnl(int(src.Lineno), "leaking param content: %v", Nconv(src, obj.FmtShort))
                                        } else {
                                                Warnl(int(src.Lineno), "leaking param content: %v level=%v dst.eld=%v src.eld=%v dst=%v",
-                                                       Nconv(src, obj.FmtShort), level, dstE.Escloopdepth, srcE.Escloopdepth, Nconv(dst, obj.FmtShort))
+                                                       Nconv(src, obj.FmtShort), level, dstE.Escloopdepth, modSrcLoopdepth, Nconv(dst, obj.FmtShort))
                                        }
                                }
                        } else {
                                                Warnl(int(src.Lineno), "leaking param: %v", Nconv(src, obj.FmtShort))
                                        } else {
                                                Warnl(int(src.Lineno), "leaking param: %v level=%v dst.eld=%v src.eld=%v dst=%v",
-                                                       Nconv(src, obj.FmtShort), level, dstE.Escloopdepth, srcE.Escloopdepth, Nconv(dst, obj.FmtShort))
+                                                       Nconv(src, obj.FmtShort), level, dstE.Escloopdepth, modSrcLoopdepth, Nconv(dst, obj.FmtShort))
                                        }
                                }
                        }
                                }
                                if Debug['m'] > 1 {
                                        Warnl(int(src.Lineno), "%v escapes to heap, level=%v, dst.eld=%v, src.eld=%v",
-                                               Nconv(p, obj.FmtShort), level, dstE.Escloopdepth, srcE.Escloopdepth)
+                                               Nconv(p, obj.FmtShort), level, dstE.Escloopdepth, modSrcLoopdepth)
                                } else {
                                        Warnl(int(src.Lineno), "%v escapes to heap", Nconv(p, obj.FmtShort))
                                }
                        }
+                       escwalkBody(e, level.dec(), dst, src.Left, modSrcLoopdepth)
+                       extraloopdepth = modSrcLoopdepth // passes to recursive case, seems likely a no-op
+               } else {
+                       escwalk(e, level.dec(), dst, src.Left)
                }
 
-               escwalk(e, level.dec(), dst, src.Left)
-
        case OAPPEND:
                escwalk(e, level, dst, src.List.N)
 
                        if Debug['m'] != 0 {
                                Warnl(int(src.Lineno), "%v escapes to heap", Nconv(src, obj.FmtShort))
                        }
+                       extraloopdepth = modSrcLoopdepth
                }
                // similar to a slice arraylit and its args.
                level = level.dec()
                        if Debug['m'] != 0 {
                                Warnl(int(src.Lineno), "%v escapes to heap", Nconv(src, obj.FmtShort))
                        }
+                       extraloopdepth = modSrcLoopdepth
                }
 
        case ODOT,
 recurse:
        level = level.copy()
        for ll := srcE.Escflowsrc; ll != nil; ll = ll.Next {
-               escwalk(e, level, dst, ll.N)
+               escwalkBody(e, level, dst, ll.N, extraloopdepth)
        }
 
        e.pdepth--
 
--- /dev/null
+// errorcheck -0 -m -l
+
+// Copyright 2015 The Go Authors.  All rights reserved.
+// Use of this source code is governed by a BSD-style
+// license that can be found in the LICENSE file.
+
+// Test, using compiler diagnostic flags, that the escape analysis is working.
+// Compiles but does not run.  Inlining is disabled.
+// Registerization is disabled too (-N), which should
+// have no effect on escape analysis.
+
+package main
+
+import "fmt"
+
+func main() {
+       // Just run test over and over again. This main func is just for
+       // convenience; if test were the main func, we could also trigger
+       // the panic just by running the program over and over again
+       // (sometimes it takes 1 time, sometimes it takes ~4,000+).
+       for iter := 0; ; iter++ {
+               if iter%50 == 0 {
+                       fmt.Println(iter) // ERROR "iter escapes to heap$" "main ... argument does not escape$"
+               }
+               test1(iter)
+               test2(iter)
+               test3(iter)
+               test4(iter)
+               test5(iter)
+               test6(iter)
+       }
+}
+
+func test1(iter int) {
+
+       const maxI = 500
+       m := make(map[int][]int) // ERROR "make\(map\[int\]\[\]int\) escapes to heap$"
+
+       // The panic seems to be triggered when m is modified inside a
+       // closure that is both recursively called and reassigned to in a
+       // loop.
+
+       // Cause of bug -- escape of closure failed to escape (shared) data structures
+       // of map.  Assign to fn declared outside of loop triggers escape of closure.
+       // Heap -> stack pointer eventually causes badness when stack reallocation
+       // occurs.
+
+       var fn func()               // ERROR "moved to heap: fn$"
+       for i := 0; i < maxI; i++ { // ERROR "moved to heap: i$"
+               // var fn func() // this makes it work, because fn stays off heap
+               j := 0        // ERROR "moved to heap: j$"
+               fn = func() { // ERROR "func literal escapes to heap$"
+                       m[i] = append(m[i], 0) // ERROR "&i escapes to heap$"
+                       if j < 25 {            // ERROR "&j escapes to heap$"
+                               j++
+                               fn() // ERROR "&fn escapes to heap$"
+                       }
+               }
+               fn()
+       }
+
+       if len(m) != maxI {
+               panic(fmt.Sprintf("iter %d: maxI = %d, len(m) = %d", iter, maxI, len(m))) // ERROR "iter escapes to heap$" "len\(m\) escapes to heap$" "maxI escapes to heap$" "test1 ... argument does not escape$"
+       }
+}
+
+func test2(iter int) {
+
+       const maxI = 500
+       m := make(map[int][]int) // ERROR "test2 make\(map\[int\]\[\]int\) does not escape$"
+
+       // var fn func()
+       for i := 0; i < maxI; i++ {
+               var fn func() // this makes it work, because fn stays off heap
+               j := 0
+               fn = func() { // ERROR "test2 func literal does not escape$"
+                       m[i] = append(m[i], 0)
+                       if j < 25 {
+                               j++
+                               fn()
+                       }
+               }
+               fn()
+       }
+
+       if len(m) != maxI {
+               panic(fmt.Sprintf("iter %d: maxI = %d, len(m) = %d", iter, maxI, len(m))) // ERROR "iter escapes to heap$" "len\(m\) escapes to heap$" "maxI escapes to heap$" "test2 ... argument does not escape$"
+       }
+}
+
+func test3(iter int) {
+
+       const maxI = 500
+       var x int // ERROR "moved to heap: x$"
+       m := &x   // ERROR "&x escapes to heap$"
+
+       var fn func() // ERROR "moved to heap: fn$"
+       for i := 0; i < maxI; i++ {
+               // var fn func() // this makes it work, because fn stays off heap
+               j := 0        // ERROR "moved to heap: j$"
+               fn = func() { // ERROR "func literal escapes to heap$"
+                       if j < 100 { // ERROR "&j escapes to heap$"
+                               j++
+                               fn() // ERROR "&fn escapes to heap$"
+                       } else {
+                               *m = *m + 1
+                       }
+               }
+               fn()
+       }
+
+       if *m != maxI {
+               panic(fmt.Sprintf("iter %d: maxI = %d, *m = %d", iter, maxI, *m)) // ERROR "\*m escapes to heap$" "iter escapes to heap$" "maxI escapes to heap$" "test3 ... argument does not escape$"
+       }
+}
+
+func test4(iter int) {
+
+       const maxI = 500
+       var x int
+       m := &x // ERROR "test4 &x does not escape$"
+
+       // var fn func()
+       for i := 0; i < maxI; i++ {
+               var fn func() // this makes it work, because fn stays off heap
+               j := 0
+               fn = func() { // ERROR "test4 func literal does not escape$"
+                       if j < 100 {
+                               j++
+                               fn()
+                       } else {
+                               *m = *m + 1
+                       }
+               }
+               fn()
+       }
+
+       if *m != maxI {
+               panic(fmt.Sprintf("iter %d: maxI = %d, *m = %d", iter, maxI, *m)) // ERROR "\*m escapes to heap$" "iter escapes to heap$" "maxI escapes to heap$" "test4 ... argument does not escape$"
+       }
+}
+
+type str struct {
+       m *int
+}
+
+func recur1(j int, s *str) { // ERROR "recur1 s does not escape"
+       if j < 100 {
+               j++
+               recur1(j, s)
+       } else {
+               *s.m++
+       }
+}
+
+func test5(iter int) {
+
+       const maxI = 500
+       var x int // ERROR "moved to heap: x$"
+       m := &x   // ERROR "&x escapes to heap$"
+
+       var fn *str
+       for i := 0; i < maxI; i++ {
+               // var fn *str // this makes it work, because fn stays off heap
+               fn = &str{m} // ERROR "&str literal escapes to heap"
+               recur1(0, fn)
+       }
+
+       if *m != maxI {
+               panic(fmt.Sprintf("iter %d: maxI = %d, *m = %d", iter, maxI, *m)) // ERROR "\*m escapes to heap$" "iter escapes to heap$" "maxI escapes to heap$" "test5 ... argument does not escape$"
+       }
+}
+
+func test6(iter int) {
+
+       const maxI = 500
+       var x int
+       m := &x // ERROR "&x does not escape$"
+
+       // var fn *str
+       for i := 0; i < maxI; i++ {
+               var fn *str  // this makes it work, because fn stays off heap
+               fn = &str{m} // ERROR "&str literal does not escape"
+               recur1(0, fn)
+       }
+
+       if *m != maxI {
+               panic(fmt.Sprintf("iter %d: maxI = %d, *m = %d", iter, maxI, *m)) // ERROR "\*m escapes to heap$" "iter escapes to heap$" "maxI escapes to heap$" "test6 ... argument does not escape$"
+       }
+}