|  | package main | 
|  |  | 
|  | import ( | 
|  | "fmt" | 
|  | ) | 
|  |  | 
|  | // Tests of call-graph queries. | 
|  | // See go.tools/guru/guru_test.go for explanation. | 
|  | // See calls.golden for expected query results. | 
|  |  | 
|  | func A(x *int) { // @pointsto pointsto-A-x "x" | 
|  | // @callers callers-A "^" | 
|  | // @callstack callstack-A "^" | 
|  | } | 
|  |  | 
|  | func B(x *int) { // @pointsto pointsto-B-x "x" | 
|  | // @callers callers-B "^" | 
|  | } | 
|  |  | 
|  | func foo() { | 
|  | } | 
|  |  | 
|  | // apply is not (yet) treated context-sensitively. | 
|  | func apply(f func(x *int), x *int) { | 
|  | f(x) // @callees callees-apply "f" | 
|  | // @callers callers-apply "^" | 
|  | } | 
|  |  | 
|  | // store *is* treated context-sensitively, | 
|  | // so the points-to sets for pc, pd are precise. | 
|  | func store(ptr **int, value *int) { | 
|  | *ptr = value | 
|  | // @callers callers-store "^" | 
|  | } | 
|  |  | 
|  | func call(f func() *int) { | 
|  | // Result points to anon function. | 
|  | f() // @pointsto pointsto-result-f "f" | 
|  |  | 
|  | // Target of call is anon function. | 
|  | f() // @callees callees-main.call-f "f" | 
|  |  | 
|  | // @callers callers-main.call "^" | 
|  | } | 
|  |  | 
|  | func main() { | 
|  | var a, b int | 
|  | go apply(A, &a) // @callees callees-main-apply1 "app" | 
|  | defer apply(B, &b) | 
|  |  | 
|  | var c, d int | 
|  | var pc, pd *int // @pointsto pointsto-pc "pc" | 
|  | store(&pc, &c) | 
|  | store(&pd, &d) | 
|  | _ = pd // @pointsto pointsto-pd "pd" | 
|  |  | 
|  | call(func() *int { | 
|  | // We are called twice from main.call | 
|  | // @callers callers-main.anon "^" | 
|  | return &a | 
|  | }) | 
|  |  | 
|  | // Errors | 
|  | _ = "no function call here"   // @callees callees-err-no-call "no" | 
|  | print("builtin")              // @callees callees-err-builtin "builtin" | 
|  | _ = string("type conversion") // @callees callees-err-conversion "str" | 
|  | call(nil)                     // @callees callees-err-bad-selection "call\\(nil" | 
|  | if false { | 
|  | main() // @callees callees-err-deadcode1 "main" | 
|  | } | 
|  | var nilFunc func() | 
|  | nilFunc() // @callees callees-err-nil-func "nilFunc" | 
|  | var i interface { | 
|  | f() | 
|  | } | 
|  | i.f() // @callees callees-err-nil-interface "i.f" | 
|  |  | 
|  | i = new(myint) | 
|  | i.f() // @callees callees-not-a-wrapper "f" | 
|  |  | 
|  | // statically dispatched calls. Handled specially by callees, so test that they work. | 
|  | foo()         // @callees callees-static-call "foo" | 
|  | fmt.Println() // @callees callees-qualified-call "Println" | 
|  | m := new(method) | 
|  | m.f() // @callees callees-static-method-call "f" | 
|  | g := new(embeddedIface) | 
|  | g.iface = m | 
|  | g.f() // @callees callees-implicit-selection-method-call "f" | 
|  | } | 
|  |  | 
|  | type myint int | 
|  |  | 
|  | func (myint) f() { | 
|  | // @callers callers-not-a-wrapper "^" | 
|  | } | 
|  |  | 
|  | type method int | 
|  |  | 
|  | func (method) f() { | 
|  | } | 
|  |  | 
|  | type embeddedIface struct { | 
|  | iface | 
|  | } | 
|  |  | 
|  | type iface interface { | 
|  | f() | 
|  | } | 
|  |  | 
|  | var dynamic = func() {} | 
|  |  | 
|  | func deadcode() { | 
|  | main() // @callees callees-err-deadcode2 "main" | 
|  | // @callers callers-err-deadcode "^" | 
|  | // @callstack callstack-err-deadcode "^" | 
|  |  | 
|  | // Within dead code, dynamic calls have no callees. | 
|  | dynamic() // @callees callees-err-deadcode3 "dynamic" | 
|  | } | 
|  |  | 
|  | // This code belongs to init. | 
|  | var global = 123 // @callers callers-global "global" | 
|  |  | 
|  | // The package initializer may be called by other packages' inits, or | 
|  | // in this case, the root of the callgraph.  The source-level init functions | 
|  | // are in turn called by it. | 
|  | func init() { | 
|  | // @callstack callstack-init "^" | 
|  | } |