...

Source file src/go/types/scope.go

Documentation: go/types

		 1  // Copyright 2013 The Go Authors. All rights reserved.
		 2  // Use of this source code is governed by a BSD-style
		 3  // license that can be found in the LICENSE file.
		 4  
		 5  // This file implements Scopes.
		 6  
		 7  package types
		 8  
		 9  import (
		10  	"bytes"
		11  	"fmt"
		12  	"go/token"
		13  	"io"
		14  	"sort"
		15  	"strings"
		16  )
		17  
		18  // A Scope maintains a set of objects and links to its containing
		19  // (parent) and contained (children) scopes. Objects may be inserted
		20  // and looked up by name. The zero value for Scope is a ready-to-use
		21  // empty scope.
		22  type Scope struct {
		23  	parent	 *Scope
		24  	children []*Scope
		25  	elems		map[string]Object // lazily allocated
		26  	pos, end token.Pos				 // scope extent; may be invalid
		27  	comment	string						// for debugging only
		28  	isFunc	 bool							// set if this is a function scope (internal use only)
		29  }
		30  
		31  // NewScope returns a new, empty scope contained in the given parent
		32  // scope, if any. The comment is for debugging only.
		33  func NewScope(parent *Scope, pos, end token.Pos, comment string) *Scope {
		34  	s := &Scope{parent, nil, nil, pos, end, comment, false}
		35  	// don't add children to Universe scope!
		36  	if parent != nil && parent != Universe {
		37  		parent.children = append(parent.children, s)
		38  	}
		39  	return s
		40  }
		41  
		42  // Parent returns the scope's containing (parent) scope.
		43  func (s *Scope) Parent() *Scope { return s.parent }
		44  
		45  // Len returns the number of scope elements.
		46  func (s *Scope) Len() int { return len(s.elems) }
		47  
		48  // Names returns the scope's element names in sorted order.
		49  func (s *Scope) Names() []string {
		50  	names := make([]string, len(s.elems))
		51  	i := 0
		52  	for name := range s.elems {
		53  		names[i] = name
		54  		i++
		55  	}
		56  	sort.Strings(names)
		57  	return names
		58  }
		59  
		60  // NumChildren returns the number of scopes nested in s.
		61  func (s *Scope) NumChildren() int { return len(s.children) }
		62  
		63  // Child returns the i'th child scope for 0 <= i < NumChildren().
		64  func (s *Scope) Child(i int) *Scope { return s.children[i] }
		65  
		66  // Lookup returns the object in scope s with the given name if such an
		67  // object exists; otherwise the result is nil.
		68  func (s *Scope) Lookup(name string) Object {
		69  	return s.elems[name]
		70  }
		71  
		72  // LookupParent follows the parent chain of scopes starting with s until
		73  // it finds a scope where Lookup(name) returns a non-nil object, and then
		74  // returns that scope and object. If a valid position pos is provided,
		75  // only objects that were declared at or before pos are considered.
		76  // If no such scope and object exists, the result is (nil, nil).
		77  //
		78  // Note that obj.Parent() may be different from the returned scope if the
		79  // object was inserted into the scope and already had a parent at that
		80  // time (see Insert). This can only happen for dot-imported objects
		81  // whose scope is the scope of the package that exported them.
		82  func (s *Scope) LookupParent(name string, pos token.Pos) (*Scope, Object) {
		83  	for ; s != nil; s = s.parent {
		84  		if obj := s.elems[name]; obj != nil && (!pos.IsValid() || obj.scopePos() <= pos) {
		85  			return s, obj
		86  		}
		87  	}
		88  	return nil, nil
		89  }
		90  
		91  // Insert attempts to insert an object obj into scope s.
		92  // If s already contains an alternative object alt with
		93  // the same name, Insert leaves s unchanged and returns alt.
		94  // Otherwise it inserts obj, sets the object's parent scope
		95  // if not already set, and returns nil.
		96  func (s *Scope) Insert(obj Object) Object {
		97  	name := obj.Name()
		98  	if alt := s.elems[name]; alt != nil {
		99  		return alt
	 100  	}
	 101  	if s.elems == nil {
	 102  		s.elems = make(map[string]Object)
	 103  	}
	 104  	s.elems[name] = obj
	 105  	if obj.Parent() == nil {
	 106  		obj.setParent(s)
	 107  	}
	 108  	return nil
	 109  }
	 110  
	 111  // squash merges s with its parent scope p by adding all
	 112  // objects of s to p, adding all children of s to the
	 113  // children of p, and removing s from p's children.
	 114  // The function f is called for each object obj in s which
	 115  // has an object alt in p. s should be discarded after
	 116  // having been squashed.
	 117  func (s *Scope) squash(err func(obj, alt Object)) {
	 118  	p := s.parent
	 119  	assert(p != nil)
	 120  	for _, obj := range s.elems {
	 121  		obj.setParent(nil)
	 122  		if alt := p.Insert(obj); alt != nil {
	 123  			err(obj, alt)
	 124  		}
	 125  	}
	 126  
	 127  	j := -1 // index of s in p.children
	 128  	for i, ch := range p.children {
	 129  		if ch == s {
	 130  			j = i
	 131  			break
	 132  		}
	 133  	}
	 134  	assert(j >= 0)
	 135  	k := len(p.children) - 1
	 136  	p.children[j] = p.children[k]
	 137  	p.children = p.children[:k]
	 138  
	 139  	p.children = append(p.children, s.children...)
	 140  
	 141  	s.children = nil
	 142  	s.elems = nil
	 143  }
	 144  
	 145  // Pos and End describe the scope's source code extent [pos, end).
	 146  // The results are guaranteed to be valid only if the type-checked
	 147  // AST has complete position information. The extent is undefined
	 148  // for Universe and package scopes.
	 149  func (s *Scope) Pos() token.Pos { return s.pos }
	 150  func (s *Scope) End() token.Pos { return s.end }
	 151  
	 152  // Contains reports whether pos is within the scope's extent.
	 153  // The result is guaranteed to be valid only if the type-checked
	 154  // AST has complete position information.
	 155  func (s *Scope) Contains(pos token.Pos) bool {
	 156  	return s.pos <= pos && pos < s.end
	 157  }
	 158  
	 159  // Innermost returns the innermost (child) scope containing
	 160  // pos. If pos is not within any scope, the result is nil.
	 161  // The result is also nil for the Universe scope.
	 162  // The result is guaranteed to be valid only if the type-checked
	 163  // AST has complete position information.
	 164  func (s *Scope) Innermost(pos token.Pos) *Scope {
	 165  	// Package scopes do not have extents since they may be
	 166  	// discontiguous, so iterate over the package's files.
	 167  	if s.parent == Universe {
	 168  		for _, s := range s.children {
	 169  			if inner := s.Innermost(pos); inner != nil {
	 170  				return inner
	 171  			}
	 172  		}
	 173  	}
	 174  
	 175  	if s.Contains(pos) {
	 176  		for _, s := range s.children {
	 177  			if s.Contains(pos) {
	 178  				return s.Innermost(pos)
	 179  			}
	 180  		}
	 181  		return s
	 182  	}
	 183  	return nil
	 184  }
	 185  
	 186  // WriteTo writes a string representation of the scope to w,
	 187  // with the scope elements sorted by name.
	 188  // The level of indentation is controlled by n >= 0, with
	 189  // n == 0 for no indentation.
	 190  // If recurse is set, it also writes nested (children) scopes.
	 191  func (s *Scope) WriteTo(w io.Writer, n int, recurse bool) {
	 192  	const ind = ".	"
	 193  	indn := strings.Repeat(ind, n)
	 194  
	 195  	fmt.Fprintf(w, "%s%s scope %p {\n", indn, s.comment, s)
	 196  
	 197  	indn1 := indn + ind
	 198  	for _, name := range s.Names() {
	 199  		fmt.Fprintf(w, "%s%s\n", indn1, s.elems[name])
	 200  	}
	 201  
	 202  	if recurse {
	 203  		for _, s := range s.children {
	 204  			s.WriteTo(w, n+1, recurse)
	 205  		}
	 206  	}
	 207  
	 208  	fmt.Fprintf(w, "%s}\n", indn)
	 209  }
	 210  
	 211  // String returns a string representation of the scope, for debugging.
	 212  func (s *Scope) String() string {
	 213  	var buf bytes.Buffer
	 214  	s.WriteTo(&buf, 0, false)
	 215  	return buf.String()
	 216  }
	 217  

View as plain text