Since this is not mentioned in the language specification, this is an implementation detail, and therefore it can vary depending on several things (Go version, target OS, architecture, etc.).
, cmd/compile/internal/gc.
escape-, , , cmd/compile/internal/gc/esc.go. " " esc():
func esc(e *EscState, n *Node, up *Node) {
if n.Esc != EscHeap && n.Type != nil &&
(n.Type.Width > MaxStackVarSize ||
(n.Op == ONEW || n.Op == OPTRLIT) && n.Type.Elem().Width >= 1<<16 ||
n.Op == OMAKESLICE && !isSmallMakeSlice(n)) {
if Debug['m'] > 2 {
Warnl(n.Lineno, "%v is too large for stack", n)
}
n.Esc = EscHeap
addrescapes(n)
escassignSinkNilWhy(e, n, n, "too large for stack")
}
}
, , isSmallMakeSlice(), cmd/compile/internal/gc/walk.go:
func isSmallMakeSlice(n *Node) bool {
if n.Op != OMAKESLICE {
return false
}
l := n.Left
r := n.Right
if r == nil {
r = l
}
t := n.Type
return Smallintconst(l) && Smallintconst(r) && (t.Elem().Width == 0 || r.Int64() < (1<<16)/t.Elem().Width)
}
:
r.Int64() < (1<<16)/t.Elem().Width
r - ( ), t.Elem().Width - :
NumElem < 65536 / SizeElem
:
NumElem < 65536 / 8 = 8192
, []uint64, 8192 - , ( ), .