2019-11-07 19:05:39 +00:00
|
|
|
// Copyright 2013 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.
|
|
|
|
|
2020-02-26 18:45:19 +00:00
|
|
|
package ir
|
2019-11-07 19:05:39 +00:00
|
|
|
|
|
|
|
// This file defines the Const SSA value type.
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"go/constant"
|
|
|
|
"go/types"
|
|
|
|
"strconv"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewConst returns a new constant of the specified value and type.
|
|
|
|
// val must be valid according to the specification of Const.Value.
|
|
|
|
//
|
|
|
|
func NewConst(val constant.Value, typ types.Type) *Const {
|
2020-02-26 18:45:19 +00:00
|
|
|
return &Const{
|
|
|
|
register: register{
|
|
|
|
typ: typ,
|
|
|
|
},
|
|
|
|
Value: val,
|
|
|
|
}
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// intConst returns an 'int' constant that evaluates to i.
|
|
|
|
// (i is an int64 in case the host is narrower than the target.)
|
|
|
|
func intConst(i int64) *Const {
|
|
|
|
return NewConst(constant.MakeInt64(i), tInt)
|
|
|
|
}
|
|
|
|
|
|
|
|
// nilConst returns a nil constant of the specified type, which may
|
|
|
|
// be any reference type, including interfaces.
|
|
|
|
//
|
|
|
|
func nilConst(typ types.Type) *Const {
|
|
|
|
return NewConst(nil, typ)
|
|
|
|
}
|
|
|
|
|
|
|
|
// stringConst returns a 'string' constant that evaluates to s.
|
|
|
|
func stringConst(s string) *Const {
|
|
|
|
return NewConst(constant.MakeString(s), tString)
|
|
|
|
}
|
|
|
|
|
|
|
|
// zeroConst returns a new "zero" constant of the specified type,
|
|
|
|
// which must not be an array or struct type: the zero values of
|
|
|
|
// aggregates are well-defined but cannot be represented by Const.
|
|
|
|
//
|
|
|
|
func zeroConst(t types.Type) *Const {
|
|
|
|
switch t := t.(type) {
|
|
|
|
case *types.Basic:
|
|
|
|
switch {
|
|
|
|
case t.Info()&types.IsBoolean != 0:
|
|
|
|
return NewConst(constant.MakeBool(false), t)
|
|
|
|
case t.Info()&types.IsNumeric != 0:
|
|
|
|
return NewConst(constant.MakeInt64(0), t)
|
|
|
|
case t.Info()&types.IsString != 0:
|
|
|
|
return NewConst(constant.MakeString(""), t)
|
|
|
|
case t.Kind() == types.UnsafePointer:
|
|
|
|
fallthrough
|
|
|
|
case t.Kind() == types.UntypedNil:
|
|
|
|
return nilConst(t)
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprint("zeroConst for unexpected type:", t))
|
|
|
|
}
|
|
|
|
case *types.Pointer, *types.Slice, *types.Interface, *types.Chan, *types.Map, *types.Signature:
|
|
|
|
return nilConst(t)
|
|
|
|
case *types.Named:
|
|
|
|
return NewConst(zeroConst(t.Underlying()).Value, t)
|
|
|
|
case *types.Array, *types.Struct, *types.Tuple:
|
|
|
|
panic(fmt.Sprint("zeroConst applied to aggregate:", t))
|
|
|
|
}
|
|
|
|
panic(fmt.Sprint("zeroConst: unexpected ", t))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Const) RelString(from *types.Package) string {
|
2020-02-26 18:45:19 +00:00
|
|
|
var p string
|
2019-11-07 19:05:39 +00:00
|
|
|
if c.Value == nil {
|
2020-02-26 18:45:19 +00:00
|
|
|
p = "nil"
|
2019-11-07 19:05:39 +00:00
|
|
|
} else if c.Value.Kind() == constant.String {
|
2020-02-26 18:45:19 +00:00
|
|
|
v := constant.StringVal(c.Value)
|
2019-11-07 19:05:39 +00:00
|
|
|
const max = 20
|
|
|
|
// TODO(adonovan): don't cut a rune in half.
|
2020-02-26 18:45:19 +00:00
|
|
|
if len(v) > max {
|
|
|
|
v = v[:max-3] + "..." // abbreviate
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
2020-02-26 18:45:19 +00:00
|
|
|
p = strconv.Quote(v)
|
2019-11-07 19:05:39 +00:00
|
|
|
} else {
|
2020-02-26 18:45:19 +00:00
|
|
|
p = c.Value.String()
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
2020-02-26 18:45:19 +00:00
|
|
|
return fmt.Sprintf("Const <%s> {%s}", relType(c.Type(), from), p)
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Const) String() string {
|
2020-02-26 18:45:19 +00:00
|
|
|
return c.RelString(c.Parent().pkg())
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsNil returns true if this constant represents a typed or untyped nil value.
|
|
|
|
func (c *Const) IsNil() bool {
|
|
|
|
return c.Value == nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Int64 returns the numeric value of this constant truncated to fit
|
|
|
|
// a signed 64-bit integer.
|
|
|
|
//
|
|
|
|
func (c *Const) Int64() int64 {
|
|
|
|
switch x := constant.ToInt(c.Value); x.Kind() {
|
|
|
|
case constant.Int:
|
|
|
|
if i, ok := constant.Int64Val(x); ok {
|
|
|
|
return i
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
case constant.Float:
|
|
|
|
f, _ := constant.Float64Val(x)
|
|
|
|
return int64(f)
|
|
|
|
}
|
|
|
|
panic(fmt.Sprintf("unexpected constant value: %T", c.Value))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Uint64 returns the numeric value of this constant truncated to fit
|
|
|
|
// an unsigned 64-bit integer.
|
|
|
|
//
|
|
|
|
func (c *Const) Uint64() uint64 {
|
|
|
|
switch x := constant.ToInt(c.Value); x.Kind() {
|
|
|
|
case constant.Int:
|
|
|
|
if u, ok := constant.Uint64Val(x); ok {
|
|
|
|
return u
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
case constant.Float:
|
|
|
|
f, _ := constant.Float64Val(x)
|
|
|
|
return uint64(f)
|
|
|
|
}
|
|
|
|
panic(fmt.Sprintf("unexpected constant value: %T", c.Value))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Float64 returns the numeric value of this constant truncated to fit
|
|
|
|
// a float64.
|
|
|
|
//
|
|
|
|
func (c *Const) Float64() float64 {
|
|
|
|
f, _ := constant.Float64Val(c.Value)
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
|
|
|
// Complex128 returns the complex value of this constant truncated to
|
|
|
|
// fit a complex128.
|
|
|
|
//
|
|
|
|
func (c *Const) Complex128() complex128 {
|
|
|
|
re, _ := constant.Float64Val(constant.Real(c.Value))
|
|
|
|
im, _ := constant.Float64Val(constant.Imag(c.Value))
|
|
|
|
return complex(re, im)
|
|
|
|
}
|