353 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			353 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Go
		
	
	
	
| // 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.
 | ||
| 
 | ||
| package interp
 | ||
| 
 | ||
| // Emulated functions that we cannot interpret because they are
 | ||
| // external or because they use "unsafe" or "reflect" operations.
 | ||
| 
 | ||
| import (
 | ||
| 	"math"
 | ||
| 	"os"
 | ||
| 	"runtime"
 | ||
| 	"syscall"
 | ||
| 	"time"
 | ||
| 
 | ||
| 	"code.google.com/p/go.tools/ssa"
 | ||
| )
 | ||
| 
 | ||
| type externalFn func(fn *ssa.Function, args []value) value
 | ||
| 
 | ||
| // TODO(adonovan): fix: reflect.Value abstracts an lvalue or an
 | ||
| // rvalue; Set() causes mutations that can be observed via aliases.
 | ||
| // We have not captured that correctly here.
 | ||
| 
 | ||
| // Key strings are from Function.FullName().
 | ||
| // That little dot ۰ is an Arabic zero numeral (U+06F0), categories [Nd].
 | ||
| var externals = map[string]externalFn{
 | ||
| 	"(reflect.Value).Bool":            ext۰reflect۰Value۰Bool,
 | ||
| 	"(reflect.Value).CanAddr":         ext۰reflect۰Value۰CanAddr,
 | ||
| 	"(reflect.Value).CanInterface":    ext۰reflect۰Value۰CanInterface,
 | ||
| 	"(reflect.Value).Elem":            ext۰reflect۰Value۰Elem,
 | ||
| 	"(reflect.Value).Field":           ext۰reflect۰Value۰Field,
 | ||
| 	"(reflect.Value).Index":           ext۰reflect۰Value۰Index,
 | ||
| 	"(reflect.Value).Int":             ext۰reflect۰Value۰Int,
 | ||
| 	"(reflect.Value).Interface":       ext۰reflect۰Value۰Interface,
 | ||
| 	"(reflect.Value).IsNil":           ext۰reflect۰Value۰IsNil,
 | ||
| 	"(reflect.Value).IsValid":         ext۰reflect۰Value۰IsValid,
 | ||
| 	"(reflect.Value).Kind":            ext۰reflect۰Value۰Kind,
 | ||
| 	"(reflect.Value).Len":             ext۰reflect۰Value۰Len,
 | ||
| 	"(reflect.Value).NumField":        ext۰reflect۰Value۰NumField,
 | ||
| 	"(reflect.Value).Pointer":         ext۰reflect۰Value۰Pointer,
 | ||
| 	"(reflect.Value).String":          ext۰reflect۰Value۰String,
 | ||
| 	"(reflect.Value).Type":            ext۰reflect۰Value۰Type,
 | ||
| 	"(reflect.error).Error":           ext۰reflect۰error۰Error,
 | ||
| 	"(reflect.rtype).Bits":            ext۰reflect۰rtype۰Bits,
 | ||
| 	"(reflect.rtype).Elem":            ext۰reflect۰rtype۰Elem,
 | ||
| 	"(reflect.rtype).Kind":            ext۰reflect۰rtype۰Kind,
 | ||
| 	"(reflect.rtype).NumOut":          ext۰reflect۰rtype۰NumOut,
 | ||
| 	"(reflect.rtype).Out":             ext۰reflect۰rtype۰Out,
 | ||
| 	"(reflect.rtype).String":          ext۰reflect۰rtype۰String,
 | ||
| 	"bytes.Equal":                     ext۰bytes۰Equal,
 | ||
| 	"bytes.IndexByte":                 ext۰bytes۰IndexByte,
 | ||
| 	"math.Float32bits":                ext۰math۰Float32bits,
 | ||
| 	"math.Float32frombits":            ext۰math۰Float32frombits,
 | ||
| 	"math.Float64bits":                ext۰math۰Float64bits,
 | ||
| 	"math.Float64frombits":            ext۰math۰Float64frombits,
 | ||
| 	"reflect.TypeOf":                  ext۰reflect۰TypeOf,
 | ||
| 	"reflect.ValueOf":                 ext۰reflect۰ValueOf,
 | ||
| 	"reflect.init":                    ext۰reflect۰Init,
 | ||
| 	"reflect.valueInterface":          ext۰reflect۰valueInterface,
 | ||
| 	"runtime.Breakpoint":              ext۰runtime۰Breakpoint,
 | ||
| 	"runtime.GC":                      ext۰runtime۰GC,
 | ||
| 	"runtime.GOMAXPROCS":              ext۰runtime۰GOMAXPROCS,
 | ||
| 	"runtime.Gosched":                 ext۰runtime۰Gosched,
 | ||
| 	"runtime.NumCPU":                  ext۰runtime۰NumCPU,
 | ||
| 	"runtime.ReadMemStats":            ext۰runtime۰ReadMemStats,
 | ||
| 	"runtime.SetFinalizer":            ext۰runtime۰SetFinalizer,
 | ||
| 	"runtime.getgoroot":               ext۰runtime۰getgoroot,
 | ||
| 	"sync/atomic.AddInt32":            ext۰atomic۰AddInt32,
 | ||
| 	"sync/atomic.CompareAndSwapInt32": ext۰atomic۰CompareAndSwapInt32,
 | ||
| 	"sync/atomic.LoadInt32":           ext۰atomic۰LoadInt32,
 | ||
| 	"sync/atomic.LoadUint32":          ext۰atomic۰LoadUint32,
 | ||
| 	"sync/atomic.StoreInt32":          ext۰atomic۰StoreInt32,
 | ||
| 	"sync/atomic.StoreUint32":         ext۰atomic۰StoreUint32,
 | ||
| 	"syscall.Close":                   ext۰syscall۰Close,
 | ||
| 	"syscall.Exit":                    ext۰syscall۰Exit,
 | ||
| 	"syscall.Fstat":                   ext۰syscall۰Fstat,
 | ||
| 	"syscall.Getpid":                  ext۰syscall۰Getpid,
 | ||
| 	"syscall.Getwd":                   ext۰syscall۰Getwd,
 | ||
| 	"syscall.Kill":                    ext۰syscall۰Kill,
 | ||
| 	"syscall.Lstat":                   ext۰syscall۰Lstat,
 | ||
| 	"syscall.Open":                    ext۰syscall۰Open,
 | ||
| 	"syscall.ParseDirent":             ext۰syscall۰ParseDirent,
 | ||
| 	"syscall.Read":                    ext۰syscall۰Read,
 | ||
| 	"syscall.ReadDirent":              ext۰syscall۰ReadDirent,
 | ||
| 	"syscall.Stat":                    ext۰syscall۰Stat,
 | ||
| 	"syscall.Write":                   ext۰syscall۰Write,
 | ||
| 	"time.Sleep":                      ext۰time۰Sleep,
 | ||
| 	"time.now":                        ext۰time۰now,
 | ||
| }
 | ||
| 
 | ||
| // wrapError returns an interpreted 'error' interface value for err.
 | ||
| func wrapError(err error) value {
 | ||
| 	if err == nil {
 | ||
| 		return iface{}
 | ||
| 	}
 | ||
| 	return iface{t: errorType, v: err.Error()}
 | ||
| }
 | ||
| 
 | ||
| func ext۰bytes۰Equal(fn *ssa.Function, args []value) value {
 | ||
| 	// func Equal(a, b []byte) bool
 | ||
| 	a := args[0].([]value)
 | ||
| 	b := args[1].([]value)
 | ||
| 	if len(a) != len(b) {
 | ||
| 		return false
 | ||
| 	}
 | ||
| 	for i := range a {
 | ||
| 		if a[i] != b[i] {
 | ||
| 			return false
 | ||
| 		}
 | ||
| 	}
 | ||
| 	return true
 | ||
| }
 | ||
| 
 | ||
| func ext۰bytes۰IndexByte(fn *ssa.Function, args []value) value {
 | ||
| 	// func IndexByte(s []byte, c byte) int
 | ||
| 	s := args[0].([]value)
 | ||
| 	c := args[1].(byte)
 | ||
| 	for i, b := range s {
 | ||
| 		if b.(byte) == c {
 | ||
| 			return i
 | ||
| 		}
 | ||
| 	}
 | ||
| 	return -1
 | ||
| }
 | ||
| 
 | ||
| func ext۰math۰Float64frombits(fn *ssa.Function, args []value) value {
 | ||
| 	return math.Float64frombits(args[0].(uint64))
 | ||
| }
 | ||
| 
 | ||
| func ext۰math۰Float64bits(fn *ssa.Function, args []value) value {
 | ||
| 	return math.Float64bits(args[0].(float64))
 | ||
| }
 | ||
| 
 | ||
| func ext۰math۰Float32frombits(fn *ssa.Function, args []value) value {
 | ||
| 	return math.Float32frombits(args[0].(uint32))
 | ||
| }
 | ||
| 
 | ||
| func ext۰math۰Float32bits(fn *ssa.Function, args []value) value {
 | ||
| 	return math.Float32bits(args[0].(float32))
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰Breakpoint(fn *ssa.Function, args []value) value {
 | ||
| 	runtime.Breakpoint()
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰getgoroot(fn *ssa.Function, args []value) value {
 | ||
| 	return os.Getenv("GOROOT")
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰GOMAXPROCS(fn *ssa.Function, args []value) value {
 | ||
| 	return runtime.GOMAXPROCS(args[0].(int))
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰GC(fn *ssa.Function, args []value) value {
 | ||
| 	runtime.GC()
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰Gosched(fn *ssa.Function, args []value) value {
 | ||
| 	runtime.Gosched()
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰NumCPU(fn *ssa.Function, args []value) value {
 | ||
| 	return runtime.NumCPU()
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰ReadMemStats(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): populate args[0].(Struct)
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰atomic۰LoadUint32(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): fix: not atomic!
 | ||
| 	return (*args[0].(*value)).(uint32)
 | ||
| }
 | ||
| 
 | ||
| func ext۰atomic۰StoreUint32(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): fix: not atomic!
 | ||
| 	*args[0].(*value) = args[1].(uint32)
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰atomic۰LoadInt32(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): fix: not atomic!
 | ||
| 	return (*args[0].(*value)).(int32)
 | ||
| }
 | ||
| 
 | ||
| func ext۰atomic۰StoreInt32(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): fix: not atomic!
 | ||
| 	*args[0].(*value) = args[1].(int32)
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰atomic۰CompareAndSwapInt32(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): fix: not atomic!
 | ||
| 	p := args[0].(*value)
 | ||
| 	if (*p).(int32) == args[1].(int32) {
 | ||
| 		*p = args[2].(int32)
 | ||
| 		return true
 | ||
| 	}
 | ||
| 	return false
 | ||
| }
 | ||
| 
 | ||
| func ext۰atomic۰AddInt32(fn *ssa.Function, args []value) value {
 | ||
| 	// TODO(adonovan): fix: not atomic!
 | ||
| 	p := args[0].(*value)
 | ||
| 	newv := (*p).(int32) + args[1].(int32)
 | ||
| 	*p = newv
 | ||
| 	return newv
 | ||
| }
 | ||
| 
 | ||
| func ext۰runtime۰SetFinalizer(fn *ssa.Function, args []value) value {
 | ||
| 	return nil // ignore
 | ||
| }
 | ||
| 
 | ||
| func ext۰time۰now(fn *ssa.Function, args []value) value {
 | ||
| 	nano := time.Now().UnixNano()
 | ||
| 	return tuple{int64(nano / 1e9), int32(nano % 1e9)}
 | ||
| }
 | ||
| 
 | ||
| func ext۰time۰Sleep(fn *ssa.Function, args []value) value {
 | ||
| 	time.Sleep(time.Duration(args[0].(int64)))
 | ||
| 	return nil
 | ||
| }
 | ||
| 
 | ||
| func ext۰syscall۰Exit(fn *ssa.Function, args []value) value {
 | ||
| 	panic(exitPanic(args[0].(int)))
 | ||
| }
 | ||
| 
 | ||
| func ext۰syscall۰Getwd(fn *ssa.Function, args []value) value {
 | ||
| 	s, err := syscall.Getwd()
 | ||
| 	return tuple{s, wrapError(err)}
 | ||
| }
 | ||
| 
 | ||
| func ext۰syscall۰Getpid(fn *ssa.Function, args []value) value {
 | ||
| 	return syscall.Getpid()
 | ||
| }
 | ||
| 
 | ||
| // The set of remaining native functions we need to implement (as needed):
 | ||
| 
 | ||
| // crypto/aes/cipher_asm.go:10:func hasAsm() bool
 | ||
| // crypto/aes/cipher_asm.go:11:func encryptBlockAsm(nr int, xk *uint32, dst, src *byte)
 | ||
| // crypto/aes/cipher_asm.go:12:func decryptBlockAsm(nr int, xk *uint32, dst, src *byte)
 | ||
| // crypto/aes/cipher_asm.go:13:func expandKeyAsm(nr int, key *byte, enc *uint32, dec *uint32)
 | ||
| // hash/crc32/crc32_amd64.go:12:func haveSSE42() bool
 | ||
| // hash/crc32/crc32_amd64.go:16:func castagnoliSSE42(crc uint32, p []byte) uint32
 | ||
| // math/abs.go:12:func Abs(x float64) float64
 | ||
| // math/asin.go:19:func Asin(x float64) float64
 | ||
| // math/asin.go:51:func Acos(x float64) float64
 | ||
| // math/atan.go:95:func Atan(x float64) float64
 | ||
| // math/atan2.go:29:func Atan2(y, x float64) float64
 | ||
| // math/big/arith_decl.go:8:func mulWW(x, y Word) (z1, z0 Word)
 | ||
| // math/big/arith_decl.go:9:func divWW(x1, x0, y Word) (q, r Word)
 | ||
| // math/big/arith_decl.go:10:func addVV(z, x, y []Word) (c Word)
 | ||
| // math/big/arith_decl.go:11:func subVV(z, x, y []Word) (c Word)
 | ||
| // math/big/arith_decl.go:12:func addVW(z, x []Word, y Word) (c Word)
 | ||
| // math/big/arith_decl.go:13:func subVW(z, x []Word, y Word) (c Word)
 | ||
| // math/big/arith_decl.go:14:func shlVU(z, x []Word, s uint) (c Word)
 | ||
| // math/big/arith_decl.go:15:func shrVU(z, x []Word, s uint) (c Word)
 | ||
| // math/big/arith_decl.go:16:func mulAddVWW(z, x []Word, y, r Word) (c Word)
 | ||
| // math/big/arith_decl.go:17:func addMulVVW(z, x []Word, y Word) (c Word)
 | ||
| // math/big/arith_decl.go:18:func divWVW(z []Word, xn Word, x []Word, y Word) (r Word)
 | ||
| // math/big/arith_decl.go:19:func bitLen(x Word) (n int)
 | ||
| // math/dim.go:13:func Dim(x, y float64) float64
 | ||
| // math/dim.go:26:func Max(x, y float64) float64
 | ||
| // math/dim.go:53:func Min(x, y float64) float64
 | ||
| // math/exp.go:14:func Exp(x float64) float64
 | ||
| // math/exp.go:135:func Exp2(x float64) float64
 | ||
| // math/expm1.go:124:func Expm1(x float64) float64
 | ||
| // math/floor.go:13:func Floor(x float64) float64
 | ||
| // math/floor.go:36:func Ceil(x float64) float64
 | ||
| // math/floor.go:48:func Trunc(x float64) float64
 | ||
| // math/frexp.go:16:func Frexp(f float64) (frac float64, exp int)
 | ||
| // math/hypot.go:17:func Hypot(p, q float64) float64
 | ||
| // math/ldexp.go:14:func Ldexp(frac float64, exp int) float64
 | ||
| // math/log.go:80:func Log(x float64) float64
 | ||
| // math/log10.go:9:func Log10(x float64) float64
 | ||
| // math/log10.go:17:func Log2(x float64) float64
 | ||
| // math/log1p.go:95:func Log1p(x float64) float64
 | ||
| // math/mod.go:21:func Mod(x, y float64) float64
 | ||
| // math/modf.go:13:func Modf(f float64) (int float64, frac float64)
 | ||
| // math/remainder.go:37:func Remainder(x, y float64) float64
 | ||
| // math/sin.go:117:func Cos(x float64) float64
 | ||
| // math/sin.go:174:func Sin(x float64) float64
 | ||
| // math/sincos.go:15:func Sincos(x float64) (sin, cos float64)
 | ||
| // math/sqrt.go:14:func Sqrt(x float64) float64
 | ||
| // math/tan.go:82:func Tan(x float64) float64
 | ||
| // os/file_posix.go:14:func sigpipe() // implemented in package runtime
 | ||
| // os/signal/signal_unix.go:15:func signal_enable(uint32)
 | ||
| // os/signal/signal_unix.go:16:func signal_recv() uint32
 | ||
| // runtime/debug.go:13:func LockOSThread()
 | ||
| // runtime/debug.go:17:func UnlockOSThread()
 | ||
| // runtime/debug.go:30:func NumCgoCall() int64
 | ||
| // runtime/debug.go:33:func NumGoroutine() int
 | ||
| // runtime/debug.go:90:func MemProfile(p []MemProfileRecord, inuseZero bool) (n int, ok bool)
 | ||
| // runtime/debug.go:114:func ThreadCreateProfile(p []StackRecord) (n int, ok bool)
 | ||
| // runtime/debug.go:122:func GoroutineProfile(p []StackRecord) (n int, ok bool)
 | ||
| // runtime/debug.go:132:func CPUProfile() []byte
 | ||
| // runtime/debug.go:141:func SetCPUProfileRate(hz int)
 | ||
| // runtime/debug.go:149:func SetBlockProfileRate(rate int)
 | ||
| // runtime/debug.go:166:func BlockProfile(p []BlockProfileRecord) (n int, ok bool)
 | ||
| // runtime/debug.go:172:func Stack(buf []byte, all bool) int
 | ||
| // runtime/error.go:81:func typestring(interface{}) string
 | ||
| // runtime/extern.go:19:func Goexit()
 | ||
| // runtime/extern.go:27:func Caller(skip int) (pc uintptr, file string, line int, ok bool)
 | ||
| // runtime/extern.go:34:func Callers(skip int, pc []uintptr) int
 | ||
| // runtime/extern.go:51:func FuncForPC(pc uintptr) *Func
 | ||
| // runtime/extern.go:68:func funcline_go(*Func, uintptr) (string, int)
 | ||
| // runtime/extern.go:71:func mid() uint32
 | ||
| // runtime/pprof/pprof.go:667:func runtime_cyclesPerSecond() int64
 | ||
| // runtime/race.go:16:func RaceDisable()
 | ||
| // runtime/race.go:19:func RaceEnable()
 | ||
| // runtime/race.go:21:func RaceAcquire(addr unsafe.Pointer)
 | ||
| // runtime/race.go:22:func RaceRelease(addr unsafe.Pointer)
 | ||
| // runtime/race.go:23:func RaceReleaseMerge(addr unsafe.Pointer)
 | ||
| // runtime/race.go:25:func RaceRead(addr unsafe.Pointer)
 | ||
| // runtime/race.go:26:func RaceWrite(addr unsafe.Pointer)
 | ||
| // runtime/race.go:28:func RaceSemacquire(s *uint32)
 | ||
| // runtime/race.go:29:func RaceSemrelease(s *uint32)
 | ||
| // sync/atomic/doc.go:49:func CompareAndSwapInt64(addr *int64, old, new int64) (swapped bool)
 | ||
| // sync/atomic/doc.go:52:func CompareAndSwapUint32(addr *uint32, old, new uint32) (swapped bool)
 | ||
| // sync/atomic/doc.go:55:func CompareAndSwapUint64(addr *uint64, old, new uint64) (swapped bool)
 | ||
| // sync/atomic/doc.go:58:func CompareAndSwapUintptr(addr *uintptr, old, new uintptr) (swapped bool)
 | ||
| // sync/atomic/doc.go:61:func CompareAndSwapPointer(addr *unsafe.Pointer, old, new unsafe.Pointer) (swapped bool)
 | ||
| // sync/atomic/doc.go:67:func AddUint32(addr *uint32, delta uint32) (new uint32)
 | ||
| // sync/atomic/doc.go:70:func AddInt64(addr *int64, delta int64) (new int64)
 | ||
| // sync/atomic/doc.go:73:func AddUint64(addr *uint64, delta uint64) (new uint64)
 | ||
| // sync/atomic/doc.go:76:func AddUintptr(addr *uintptr, delta uintptr) (new uintptr)
 | ||
| // sync/atomic/doc.go:82:func LoadInt64(addr *int64) (val int64)
 | ||
| // sync/atomic/doc.go:88:func LoadUint64(addr *uint64) (val uint64)
 | ||
| // sync/atomic/doc.go:91:func LoadUintptr(addr *uintptr) (val uintptr)
 | ||
| // sync/atomic/doc.go:94:func LoadPointer(addr *unsafe.Pointer) (val unsafe.Pointer)
 | ||
| // sync/atomic/doc.go:100:func StoreInt64(addr *int64, val int64)
 | ||
| // sync/atomic/doc.go:106:func StoreUint64(addr *uint64, val uint64)
 | ||
| // sync/atomic/doc.go:109:func StoreUintptr(addr *uintptr, val uintptr)
 | ||
| // sync/atomic/doc.go:112:func StorePointer(addr *unsafe.Pointer, val unsafe.Pointer)
 | ||
| // sync/runtime.go:12:func runtime_Semacquire(s *uint32)
 | ||
| // sync/runtime.go:18:func runtime_Semrelease(s *uint32)
 | ||
| // syscall/env_unix.go:30:func setenv_c(k, v string)
 | ||
| // syscall/syscall_linux_amd64.go:60:func Gettimeofday(tv *Timeval) (err error)
 | ||
| // syscall/syscall_linux_amd64.go:61:func Time(t *Time_t) (tt Time_t, err error)
 | ||
| // syscall/syscall_linux_arm.go:28:func Seek(fd int, offset int64, whence int) (newoffset int64, err error)
 | ||
| // time/sleep.go:25:func startTimer(*runtimeTimer)
 | ||
| // time/sleep.go:26:func stopTimer(*runtimeTimer) bool
 | ||
| // time/time.go:758:func now() (sec int64, nsec int32)
 | ||
| // unsafe/unsafe.go:27:func Sizeof(v ArbitraryType) uintptr
 | ||
| // unsafe/unsafe.go:32:func Offsetof(v ArbitraryType) uintptr
 | ||
| // unsafe/unsafe.go:37:func Alignof(v ArbitraryType) uintptr
 |