Rietveld Code Review Tool
Help | Bug tracker | Discussion group | Source code | Sign in
(277)

Unified Diff: go/types/lookup.go

Issue 10459044: code review 10459044: go.tools/go/types: handle p.x with p of type P *S (Closed)
Patch Set: diff -r c23bf80f7c2f https://code.google.com/p/go.tools Created 11 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Please Sign in to add in-line comments.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « no previous file | go/types/testdata/decls3.src » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: go/types/lookup.go
===================================================================
--- a/go/types/lookup.go
+++ b/go/types/lookup.go
@@ -32,6 +32,48 @@
// index sequence points to an ambiguous entry if it exists, or it is nil.
//
func LookupFieldOrMethod(typ Type, pkg *Package, name string) (obj Object, index []int, indirect bool) {
+ obj, index, indirect = lookupFieldOrMethod(typ, pkg, name)
+ if obj != nil {
+ return
+ }
+
+ // TODO(gri) The code below is not needed if we are looking for methods only,
+ // and it can be done always if we look for fields only. Consider
+ // providing LookupField and LookupMethod as well.
+
+ // If we didn't find anything, we still might have a field p.x as in:
+ //
+ // type S struct{ x int }
+ // func (*S) m() {}
+ // type P *S
+ // var p P
+ //
+ // which requires that we start the search with the underlying type
+ // of P (i.e., *S). We cannot do this always because we might find
+ // methods that don't exist for P but for S (e.g., m). Thus, if the
+ // result is a method we need to discard it.
+ //
+ // TODO(gri) WTF? There isn't a more direct way? Perhaps we should
+ // outlaw named types to pointer types - they are almost
+ // never what one wants, anyway.
+ if t, _ := typ.(*Named); t != nil {
+ u := t.underlying
+ if _, ok := u.(*Pointer); ok {
+ // typ is a named type with an underlying type of the form *T,
+ // start the search with the underlying type *T
+ if obj2, index2, indirect2 := lookupFieldOrMethod(u, pkg, name); obj2 != nil {
+ // only if the result is a field can we keep it
+ if _, ok := obj2.(*Field); ok {
+ return obj2, index2, indirect2
+ }
+ }
+ }
+ }
+
+ return
+}
+
+func lookupFieldOrMethod(typ Type, pkg *Package, name string) (obj Object, index []int, indirect bool) {
if name == "_" {
return // empty fields/methods are never found
}
« no previous file with comments | « no previous file | go/types/testdata/decls3.src » ('j') | no next file with comments »

Powered by Google App Engine
RSS Feeds Recent Issues | This issue
This is Rietveld f62528b