| // 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) |
| } |
| } |
| } |