| // Copyright 2016 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 gc |
| |
| import ( |
| "bytes" |
| "fmt" |
| "internal/testenv" |
| "io/ioutil" |
| "os" |
| "os/exec" |
| "path/filepath" |
| "regexp" |
| "runtime" |
| "strings" |
| "testing" |
| ) |
| |
| // TestAssembly checks to make sure the assembly generated for |
| // functions contains certain expected instructions. |
| // Note: this test will fail if -ssa=0. |
| func TestAssembly(t *testing.T) { |
| testenv.MustHaveGoBuild(t) |
| if runtime.GOOS == "windows" { |
| // TODO: remove if we can get "go tool compile -S" to work on windows. |
| t.Skipf("skipping test: recursive windows compile not working") |
| } |
| dir, err := ioutil.TempDir("", "TestAssembly") |
| if err != nil { |
| t.Fatalf("could not create directory: %v", err) |
| } |
| defer os.RemoveAll(dir) |
| |
| for _, test := range asmTests { |
| asm := compileToAsm(t, dir, test.arch, fmt.Sprintf(template, test.function)) |
| // Get rid of code for "".init. Also gets rid of type algorithms & other junk. |
| if i := strings.Index(asm, "\n\"\".init "); i >= 0 { |
| asm = asm[:i+1] |
| } |
| for _, r := range test.regexps { |
| if b, err := regexp.MatchString(r, asm); !b || err != nil { |
| t.Errorf("expected:%s\ngo:%s\nasm:%s\n", r, test.function, asm) |
| } |
| } |
| } |
| } |
| |
| // compile compiles the package pkg for architecture arch and |
| // returns the generated assembly. dir is a scratch directory. |
| func compileToAsm(t *testing.T, dir, arch, pkg string) string { |
| // Create source. |
| src := filepath.Join(dir, "test.go") |
| f, err := os.Create(src) |
| if err != nil { |
| panic(err) |
| } |
| f.Write([]byte(pkg)) |
| f.Close() |
| |
| var stdout, stderr bytes.Buffer |
| cmd := exec.Command(testenv.GoToolPath(t), "tool", "compile", "-S", "-o", filepath.Join(dir, "out.o"), src) |
| cmd.Env = mergeEnvLists([]string{"GOARCH=" + arch}, os.Environ()) |
| cmd.Stdout = &stdout |
| cmd.Stderr = &stderr |
| if err := cmd.Run(); err != nil { |
| panic(err) |
| } |
| if s := stderr.String(); s != "" { |
| panic(fmt.Errorf("Stderr = %s\nWant empty", s)) |
| } |
| return stdout.String() |
| } |
| |
| // template to convert a function to a full file |
| const template = ` |
| package main |
| %s |
| ` |
| |
| type asmTest struct { |
| // architecture to compile to |
| arch string |
| // function to compile |
| function string |
| // regexps that must match the generated assembly |
| regexps []string |
| } |
| |
| var asmTests = [...]asmTest{ |
| {"amd64", ` |
| func f(x int) int { |
| return x * 64 |
| } |
| `, |
| []string{"\tSHLQ\t\\$6,"}, |
| }, |
| {"amd64", ` |
| func f(x int) int { |
| return x * 96 |
| }`, |
| []string{"\tSHLQ\t\\$5,", "\tLEAQ\t\\(.*\\)\\(.*\\*2\\),"}, |
| }, |
| } |
| |
| // mergeEnvLists merges the two environment lists such that |
| // variables with the same name in "in" replace those in "out". |
| // This always returns a newly allocated slice. |
| func mergeEnvLists(in, out []string) []string { |
| out = append([]string(nil), out...) |
| NextVar: |
| for _, inkv := range in { |
| k := strings.SplitAfterN(inkv, "=", 2)[0] |
| for i, outkv := range out { |
| if strings.HasPrefix(outkv, k) { |
| out[i] = inkv |
| continue NextVar |
| } |
| } |
| out = append(out, inkv) |
| } |
| return out |
| } |
| |
| // TestLineNumber checks to make sure the generated assembly has line numbers |
| // see issue #16214 |
| func TestLineNumber(t *testing.T) { |
| testenv.MustHaveGoBuild(t) |
| dir, err := ioutil.TempDir("", "TestLineNumber") |
| if err != nil { |
| t.Fatalf("could not create directory: %v", err) |
| } |
| defer os.RemoveAll(dir) |
| |
| src := filepath.Join(dir, "x.go") |
| err = ioutil.WriteFile(src, []byte(issue16214src), 0644) |
| if err != nil { |
| t.Fatalf("could not write file: %v", err) |
| } |
| |
| cmd := exec.Command(testenv.GoToolPath(t), "tool", "compile", "-S", "-o", filepath.Join(dir, "out.o"), src) |
| out, err := cmd.CombinedOutput() |
| if err != nil { |
| t.Fatalf("fail to run go tool compile: %v", err) |
| } |
| |
| if strings.Contains(string(out), "unknown line number") { |
| t.Errorf("line number missing in assembly:\n%s", out) |
| } |
| } |
| |
| var issue16214src = ` |
| package main |
| |
| func Mod32(x uint32) uint32 { |
| return x % 3 // frontend rewrites it as HMUL with 2863311531, the LITERAL node has Lineno 0 |
| } |
| ` |