blob: 1e757f55ab60167272df9c88294f243774a22725 [file] [log] [blame]
// 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)
}
}
}