mirror of
https://github.com/golang/go.git
synced 2025-05-06 16:13:04 +00:00
(reflect.Value).Send (reflect.Value).TrySend (reflect.Value).Recv (reflect.Value).TryRecv (reflect.Type).ChanOf (reflect.Type).In (reflect.Type).Out reflect.Indirect reflect.MakeChan Also: - specialize genInvoke when the receiver is a reflect.Type under the assumption that there's only one possible concrete type. This makes all reflect.Type operations context-sensitive since the calls are no longer dynamic. - Rename all variables to match the actual parameter names used in the reflect API. - Add pointer.Config.Reflection flag (exposed in oracle as --reflect, default false) to enable reflection. It currently adds about 20% running time. I'll make it true after the presolver is implemented. - Simplified worklist datatype and solver main loop slightly (~10% speed improvement). - Use addLabel() utility to add a label to a PTS. (Working on my 3 yr old 2x2GHz+4GB Mac vs 8x4GHz+24GB workstation, one really notices the cost of pointer analysis. Note to self: time to implement presolver.) R=crawshaw CC=golang-dev https://golang.org/cl/13242062
79 lines
2.5 KiB
Go
79 lines
2.5 KiB
Go
// +build ignore
|
|
|
|
package main
|
|
|
|
import "reflect"
|
|
|
|
// Test of channels with reflection.
|
|
|
|
var a, b int
|
|
|
|
func chanreflect1() {
|
|
ch := make(chan *int, 0) // @line cr1make
|
|
crv := reflect.ValueOf(ch)
|
|
crv.Send(reflect.ValueOf(&a))
|
|
print(crv.Interface()) // @types chan *int
|
|
print(crv.Interface().(chan *int)) // @pointsto makechan@cr1make:12
|
|
print(<-ch) // @pointsto main.a
|
|
}
|
|
|
|
func chanreflect2() {
|
|
ch := make(chan *int, 0)
|
|
ch <- &b
|
|
crv := reflect.ValueOf(ch)
|
|
r, _ := crv.Recv()
|
|
print(r.Interface()) // @types *int
|
|
print(r.Interface().(*int)) // @pointsto main.b
|
|
}
|
|
|
|
// TODO(adonovan): the analysis can't yet take advantage of the
|
|
// ChanOf(dir) parameter so the results are less precise than they
|
|
// should be: all three directions are returned.
|
|
|
|
func chanOfRecv() {
|
|
// MakeChan(<-chan) is a no-op.
|
|
t := reflect.ChanOf(reflect.RecvDir, reflect.TypeOf(&a))
|
|
print(reflect.Zero(t).Interface()) // @types <-chan *int | chan<- *int | chan *int
|
|
print(reflect.MakeChan(t, 0).Interface().(<-chan *int)) // @pointsto
|
|
print(reflect.MakeChan(t, 0).Interface().(chan *int)) // @pointsto <alloc in reflect.MakeChan>
|
|
}
|
|
|
|
func chanOfSend() {
|
|
// MakeChan(chan<-) is a no-op.
|
|
t := reflect.ChanOf(reflect.SendDir, reflect.TypeOf(&a))
|
|
print(reflect.Zero(t).Interface()) // @types <-chan *int | chan<- *int | chan *int
|
|
print(reflect.MakeChan(t, 0).Interface().(chan<- *int)) // @pointsto
|
|
print(reflect.MakeChan(t, 0).Interface().(chan *int)) // @pointsto <alloc in reflect.MakeChan>
|
|
}
|
|
|
|
func chanOfBoth() {
|
|
t := reflect.ChanOf(reflect.BothDir, reflect.TypeOf(&a))
|
|
print(reflect.Zero(t).Interface()) // @types <-chan *int | chan<- *int | chan *int
|
|
ch := reflect.MakeChan(t, 0)
|
|
print(ch.Interface().(chan *int)) // @pointsto <alloc in reflect.MakeChan>
|
|
ch.Send(reflect.ValueOf(&b))
|
|
ch.Interface().(chan *int) <- &a
|
|
r, _ := ch.Recv()
|
|
print(r.Interface().(*int)) // @pointsto main.a | main.b
|
|
print(<-ch.Interface().(chan *int)) // @pointsto main.a | main.b
|
|
}
|
|
|
|
var unknownDir reflect.ChanDir // not a constant
|
|
|
|
func chanOfUnknown() {
|
|
// Unknown channel direction: assume all three.
|
|
// MakeChan only works on the bi-di channel type.
|
|
t := reflect.ChanOf(unknownDir, reflect.TypeOf(&a))
|
|
print(reflect.Zero(t).Interface()) // @types <-chan *int | chan<- *int | chan *int
|
|
print(reflect.MakeChan(t, 0).Interface()) // @types chan *int
|
|
}
|
|
|
|
func main() {
|
|
chanreflect1()
|
|
chanreflect2()
|
|
chanOfRecv()
|
|
chanOfSend()
|
|
chanOfBoth()
|
|
chanOfUnknown()
|
|
}
|