--- /dev/null
+// Copyright 2025 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.
+
+package big
+
+import (
+ "internal/testenv"
+ "math/rand"
+ "os/exec"
+ "strings"
+ "testing"
+)
+
+func TestEscape(t *testing.T) {
+ testenv.MustHaveGoBuild(t)
+ // The multiplication routines create many temporary Int values,
+ // expecting them to be stack-allocated. Make sure none escape to the heap.
+ out, err := exec.Command("go", "build", "-gcflags=-m").CombinedOutput()
+ if err != nil {
+ t.Fatalf("go build -gcflags=-m: %v\n%s", err, out)
+ }
+ for line := range strings.Lines(string(out)) {
+ if strings.Contains(line, "natmul.go") && strings.Contains(line, "Int") && strings.Contains(line, "escapes") {
+ t.Error(strings.TrimSpace(line))
+ }
+ }
+}
+
+func TestMulAlloc(t *testing.T) {
+ r := rand.New(rand.NewSource(1234))
+ sizes := []int{karatsubaThreshold / 2, karatsubaThreshold}
+ for _, size := range sizes {
+ x := randInt(r, uint(size))
+ y := randInt(r, uint(size))
+ z := &Int{abs: make(nat, 2*uint(size))}
+ if n := testing.AllocsPerRun(10, func() { z.Mul(x, y) }); n >= 1 {
+ t.Errorf("Mul(len %d, len %d) allocates %.2f objects", size, size, n)
+ }
+ }
+}
+
+func TestSqrAlloc(t *testing.T) {
+ r := rand.New(rand.NewSource(1234))
+ sizes := []int{basicSqrThreshold / 2, basicSqrThreshold, karatsubaSqrThreshold}
+ for _, size := range sizes {
+ x := randInt(r, uint(size))
+ z := &Int{abs: make(nat, 2*uint(size))}
+ if n := testing.AllocsPerRun(10, func() { z.Mul(x, x) }); n >= 1 {
+ t.Errorf("Mul(len %d with itself) allocates %.2f objects", size, n)
+ }
+ }
+}