go/test/convinline.go
Russ Cox bed970b3ff cmd/compile: handle integer conversions in static init inliner
Given code like

	func itou(i int) uint { return uint(i) }
	var x = itou(-1)

the static inliner from CL 450136 was rewriting the code to

	var x = uint(-1)

which is not valid Go code. Fix this by converting the
constants appropriately during inlining.

Fixes golang.org/x/image/vector test.

Change-Id: I13448df8504c6a70525b1cdc36e2c947e22cdd33
Reviewed-on: https://go-review.googlesource.com/c/go/+/451376
Auto-Submit: Russ Cox <rsc@golang.org>
Reviewed-by: Keith Randall <khr@google.com>
Run-TryBot: Russ Cox <rsc@golang.org>
Reviewed-by: Keith Randall <khr@golang.org>
TryBot-Result: Gopher Robot <gobot@golang.org>
2022-11-17 13:46:05 +00:00

194 lines
4.7 KiB
Go

// runoutput
//go:build !wasm
// Copyright 2022 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 main
import (
"bytes"
"fmt"
"math"
"math/bits"
"os"
"strconv"
"strings"
)
var types = []string{
"int",
"int8",
"int16",
"int32",
"int64",
"uint",
"uint8",
"uint16",
"uint32",
"uint64",
"uintptr",
"float32",
"float64",
}
func main() {
var prog bytes.Buffer
fmt.Fprintf(&prog, "package main\n\n")
fmt.Fprintf(&prog, "import ( \"fmt\"; \"math\" )\n")
for _, t1 := range types {
for _, t2 := range types {
fmt.Fprintf(&prog, "func %[1]s_to_%[2]s(x %[1]s) %[2]s { return %[2]s(x) }\n", t1, t2)
}
}
var outputs []string
var exprs []string
fmt.Fprintf(&prog, "var (\n")
for _, t1 := range types {
var inputs []string
switch t1 {
case "int64", "int":
if t1 == "int64" || bits.UintSize == 64 {
inputs = append(inputs, "-0x8000_0000_0000_0000", "-0x7fff_ffff_ffff_ffff", "-0x12_3456_7890", "0x12_3456_7890", "0x7fff_ffff_ffff_ffff")
}
fallthrough
case "int32":
inputs = append(inputs, "-0x8000_0000", "-0x7fff_ffff", "-0x12_3456", "0x12_3456", "0x7fff_ffff")
fallthrough
case "int16":
inputs = append(inputs, "-0x8000", "-0x7fff", "-0x1234", "0x1234", "0x7fff")
fallthrough
case "int8":
inputs = append(inputs, "-0x80", "-0x7f", "-0x12", "-1", "0", "1", "0x12", "0x7f")
case "uint64", "uint", "uintptr":
if t1 == "uint64" || bits.UintSize == 64 {
inputs = append(inputs, "0x12_3456_7890", "0x7fff_ffff_ffff_ffff", "0x8000_0000_0000_0000", "0xffff_ffff_ffff_ffff")
}
fallthrough
case "uint32":
inputs = append(inputs, "0x12_3456", "0x7fff_ffff", "0x8000_0000", "0xffff_ffff")
fallthrough
case "uint16":
inputs = append(inputs, "0x1234", "0x7fff", "0x8000", "0xffff")
fallthrough
case "uint8":
inputs = append(inputs, "0", "1", "0x12", "0x7f", "0x80", "0xff")
case "float64":
inputs = append(inputs,
"-1.79769313486231570814527423731704356798070e+308",
"-1e300",
"-1e100",
"-1e40",
"-3.5e38",
"3.5e38",
"1e40",
"1e100",
"1e300",
"1.79769313486231570814527423731704356798070e+308")
fallthrough
case "float32":
inputs = append(inputs,
"-3.40282346638528859811704183484516925440e+38",
"-1e38",
"-1.5",
"-1.401298464324817070923729583289916131280e-45",
"0",
"1.401298464324817070923729583289916131280e-45",
"1.5",
"1e38",
"3.40282346638528859811704183484516925440e+38")
}
for _, t2 := range types {
for _, x := range inputs {
code := fmt.Sprintf("%s_to_%s(%s)", t1, t2, x)
fmt.Fprintf(&prog, "\tv%d = %s\n", len(outputs), code)
exprs = append(exprs, code)
outputs = append(outputs, convert(x, t1, t2))
}
}
}
fmt.Fprintf(&prog, ")\n\n")
fmt.Fprintf(&prog, "func main() {\n\tok := true\n")
for i, out := range outputs {
fmt.Fprintf(&prog, "\tif v%d != %s { fmt.Println(%q, \"=\", v%d, \"want\", %s); ok = false }\n", i, out, exprs[i], i, out)
}
fmt.Fprintf(&prog, "\tif !ok { println(\"FAIL\") }\n")
fmt.Fprintf(&prog, "}\n")
os.Stdout.Write(prog.Bytes())
}
func convert(x, t1, t2 string) string {
if strings.HasPrefix(t1, "int") {
v, err := strconv.ParseInt(x, 0, 64)
if err != nil {
println(x, t1, t2)
panic(err)
}
return convert1(v, t2)
}
if strings.HasPrefix(t1, "uint") {
v, err := strconv.ParseUint(x, 0, 64)
if err != nil {
println(x, t1, t2)
panic(err)
}
return convert1(v, t2)
}
if strings.HasPrefix(t1, "float") {
v, err := strconv.ParseFloat(x, 64)
if err != nil {
println(x, t1, t2)
panic(err)
}
if t1 == "float32" {
v = float64(float32(v))
}
return convert1(v, t2)
}
panic(t1)
}
func convert1[T int64 | uint64 | float64](v T, t2 string) string {
switch t2 {
case "int":
return fmt.Sprintf("%s(%#x)", t2, int(v))
case "int8":
return fmt.Sprintf("%s(%#x)", t2, int8(v))
case "int16":
return fmt.Sprintf("%s(%#x)", t2, int16(v))
case "int32":
return fmt.Sprintf("%s(%#x)", t2, int32(v))
case "int64":
return fmt.Sprintf("%s(%#x)", t2, int64(v))
case "uint":
return fmt.Sprintf("%s(%#x)", t2, uint(v))
case "uint8":
return fmt.Sprintf("%s(%#x)", t2, uint8(v))
case "uint16":
return fmt.Sprintf("%s(%#x)", t2, uint16(v))
case "uint32":
return fmt.Sprintf("%s(%#x)", t2, uint32(v))
case "uint64":
return fmt.Sprintf("%s(%#x)", t2, uint64(v))
case "uintptr":
return fmt.Sprintf("%s(%#x)", t2, uintptr(v))
case "float32":
v := float32(v)
if math.IsInf(float64(v), -1) {
return "float32(math.Inf(-1))"
}
if math.IsInf(float64(v), +1) {
return "float32(math.Inf(+1))"
}
return fmt.Sprintf("%s(%v)", t2, float64(v))
case "float64":
return fmt.Sprintf("%s(%v)", t2, float64(v))
}
panic(t2)
}