mirror of
https://github.com/golang/go
synced 2024-11-22 07:14:40 -07:00
go/ast: implemented NewPackage
NewPackage creates an ast.Package node from a set of package files and resolves unresolved identifiers. Also: - Changed semantics of Scope.Insert: If an object is inserted w/o errors, the result is nil (before it was obj). - Fixed an identifier resolution bug in the parser: map keys must not be resolved. gotype runs through several go/* packages and successfully resolves all (non-field/method) identifiers. R=rog, rsc CC=golang-dev https://golang.org/cl/4298044
This commit is contained in:
parent
a7a854b82f
commit
5be77a204b
@ -165,22 +165,24 @@ func processFiles(filenames []string, allFiles bool) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
processPackage(parseFiles(token.NewFileSet(), filenames[0:i]))
|
fset := token.NewFileSet()
|
||||||
|
processPackage(fset, parseFiles(fset, filenames[0:i]))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
func processPackage(files map[string]*ast.File) {
|
// TODO(gri) Replace this with a fully functioning importer.
|
||||||
// TODO(gri) Enable this code once we have ast.NewPackage.
|
// For now a dummy importer is set up by gotype_test.go.
|
||||||
/*
|
var importer ast.Importer
|
||||||
|
|
||||||
|
func processPackage(fset *token.FileSet, files map[string]*ast.File) {
|
||||||
// make a package (resolve all identifiers)
|
// make a package (resolve all identifiers)
|
||||||
pkg, err := ast.NewPackage(files)
|
pkg, err := ast.NewPackage(fset, files, importer, universe)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
report(err)
|
report(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// TODO(gri): typecheck package
|
// TODO(gri): typecheck package
|
||||||
_ = pkg
|
_ = pkg
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -189,10 +191,74 @@ func main() {
|
|||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
if flag.NArg() == 0 {
|
if flag.NArg() == 0 {
|
||||||
processPackage(parseStdin(token.NewFileSet()))
|
fset := token.NewFileSet()
|
||||||
|
processPackage(fset, parseStdin(fset))
|
||||||
} else {
|
} else {
|
||||||
processFiles(flag.Args(), true)
|
processFiles(flag.Args(), true)
|
||||||
}
|
}
|
||||||
|
|
||||||
os.Exit(exitCode)
|
os.Exit(exitCode)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// TODO(gri) Move universe and its initialization in to the right package.
|
||||||
|
var universe *ast.Scope
|
||||||
|
|
||||||
|
func define(kind ast.ObjKind, names ...string) {
|
||||||
|
for _, name := range names {
|
||||||
|
obj := ast.NewObj(kind, name)
|
||||||
|
if universe.Insert(obj) != nil {
|
||||||
|
panic("gotype internal error: incorrect universe scope")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
universe = ast.NewScope(nil)
|
||||||
|
|
||||||
|
define(ast.Typ,
|
||||||
|
"bool",
|
||||||
|
"byte",
|
||||||
|
"complex64",
|
||||||
|
"complex128",
|
||||||
|
"float32",
|
||||||
|
"float64",
|
||||||
|
"int8",
|
||||||
|
"int16",
|
||||||
|
"int32",
|
||||||
|
"int64",
|
||||||
|
"string",
|
||||||
|
"uint8",
|
||||||
|
"uint16",
|
||||||
|
"uint32",
|
||||||
|
"uint64",
|
||||||
|
"int",
|
||||||
|
"uint",
|
||||||
|
"uintptr",
|
||||||
|
)
|
||||||
|
|
||||||
|
define(ast.Con,
|
||||||
|
"true",
|
||||||
|
"false",
|
||||||
|
"iota",
|
||||||
|
"nil",
|
||||||
|
)
|
||||||
|
|
||||||
|
define(ast.Fun,
|
||||||
|
"append",
|
||||||
|
"cap",
|
||||||
|
"close",
|
||||||
|
"complex",
|
||||||
|
"copy",
|
||||||
|
"imag",
|
||||||
|
"len",
|
||||||
|
"make",
|
||||||
|
"new",
|
||||||
|
"panic",
|
||||||
|
"print",
|
||||||
|
"println",
|
||||||
|
"real",
|
||||||
|
"recover",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
@ -5,25 +5,36 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"go/ast"
|
||||||
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
|
"path"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
func testImporter(importPath string) (string, *ast.Scope, os.Error) {
|
||||||
|
_, pkgName := path.Split(importPath) // filename is package name for std library
|
||||||
|
return pkgName, ast.NewScope(nil), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
func testDir(t *testing.T, dir, pkg string) {
|
func testDir(t *testing.T, dir, pkg string) {
|
||||||
|
exitCode = 0
|
||||||
*pkgName = pkg
|
*pkgName = pkg
|
||||||
*recursive = false
|
*recursive = false
|
||||||
|
importer = testImporter
|
||||||
processDirectory(dir)
|
processDirectory(dir)
|
||||||
if exitCode != 0 {
|
if exitCode != 0 {
|
||||||
t.Errorf("processing %d failed: exitCode = %d", dir, exitCode)
|
t.Errorf("processing %s failed: exitCode = %d", dir, exitCode)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
func Test(t *testing.T) {
|
func Test(t *testing.T) {
|
||||||
testDir(t, ".", "main")
|
|
||||||
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/ast"), "ast")
|
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/ast"), "ast")
|
||||||
|
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/token"), "scanner")
|
||||||
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/scanner"), "scanner")
|
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/scanner"), "scanner")
|
||||||
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/parser"), "parser")
|
testDir(t, filepath.Join(runtime.GOROOT(), "src/pkg/go/parser"), "parser")
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ GOFILES=\
|
|||||||
ast.go\
|
ast.go\
|
||||||
filter.go\
|
filter.go\
|
||||||
print.go\
|
print.go\
|
||||||
|
resolve.go\
|
||||||
scope.go\
|
scope.go\
|
||||||
walk.go\
|
walk.go\
|
||||||
|
|
||||||
|
@ -781,7 +781,7 @@ type (
|
|||||||
ImportSpec struct {
|
ImportSpec struct {
|
||||||
Doc *CommentGroup // associated documentation; or nil
|
Doc *CommentGroup // associated documentation; or nil
|
||||||
Name *Ident // local package name (including "."); or nil
|
Name *Ident // local package name (including "."); or nil
|
||||||
Path *BasicLit // package path
|
Path *BasicLit // import path
|
||||||
Comment *CommentGroup // line comments; or nil
|
Comment *CommentGroup // line comments; or nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -925,8 +925,9 @@ type File struct {
|
|||||||
Package token.Pos // position of "package" keyword
|
Package token.Pos // position of "package" keyword
|
||||||
Name *Ident // package name
|
Name *Ident // package name
|
||||||
Decls []Decl // top-level declarations; or nil
|
Decls []Decl // top-level declarations; or nil
|
||||||
Scope *Scope // package scope
|
Scope *Scope // package scope (this file only)
|
||||||
Unresolved []*Ident // unresolved global identifiers
|
Imports []*ImportSpec // imports in this file
|
||||||
|
Unresolved []*Ident // unresolved identifiers in this file
|
||||||
Comments []*CommentGroup // list of all comments in the source file
|
Comments []*CommentGroup // list of all comments in the source file
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -946,6 +947,7 @@ func (f *File) End() token.Pos {
|
|||||||
type Package struct {
|
type Package struct {
|
||||||
Name string // package name
|
Name string // package name
|
||||||
Scope *Scope // package scope
|
Scope *Scope // package scope
|
||||||
|
Imports map[string]*Scope // map of import path -> package scope across all files
|
||||||
Files map[string]*File // Go source files by filename
|
Files map[string]*File // Go source files by filename
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -426,5 +426,6 @@ func MergePackageFiles(pkg *Package, mode MergeMode) *File {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// TODO(gri) need to compute pkgScope and unresolved identifiers!
|
// TODO(gri) need to compute pkgScope and unresolved identifiers!
|
||||||
return &File{doc, pos, NewIdent(pkg.Name), decls, nil, nil, comments}
|
// TODO(gri) need to compute imports!
|
||||||
|
return &File{doc, pos, NewIdent(pkg.Name), decls, nil, nil, nil, comments}
|
||||||
}
|
}
|
||||||
|
188
src/pkg/go/ast/resolve.go
Normal file
188
src/pkg/go/ast/resolve.go
Normal file
@ -0,0 +1,188 @@
|
|||||||
|
// Copyright 2011 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.
|
||||||
|
|
||||||
|
// This file implements NewPackage.
|
||||||
|
|
||||||
|
package ast
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"go/scanner"
|
||||||
|
"go/token"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
type pkgBuilder struct {
|
||||||
|
scanner.ErrorVector
|
||||||
|
fset *token.FileSet
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (p *pkgBuilder) error(pos token.Pos, msg string) {
|
||||||
|
p.Error(p.fset.Position(pos), msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (p *pkgBuilder) errorf(pos token.Pos, format string, args ...interface{}) {
|
||||||
|
p.error(pos, fmt.Sprintf(format, args...))
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (p *pkgBuilder) declare(scope, altScope *Scope, obj *Object) {
|
||||||
|
alt := scope.Insert(obj)
|
||||||
|
if alt == nil && altScope != nil {
|
||||||
|
// see if there is a conflicting declaration in altScope
|
||||||
|
alt = altScope.Lookup(obj.Name)
|
||||||
|
}
|
||||||
|
if alt != nil {
|
||||||
|
prevDecl := ""
|
||||||
|
if pos := alt.Pos(); pos.IsValid() {
|
||||||
|
prevDecl = fmt.Sprintf("\n\tprevious declaration at %s", p.fset.Position(pos))
|
||||||
|
}
|
||||||
|
p.error(obj.Pos(), fmt.Sprintf("%s redeclared in this block%s", obj.Name, prevDecl))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func resolve(scope *Scope, ident *Ident) bool {
|
||||||
|
for ; scope != nil; scope = scope.Outer {
|
||||||
|
if obj := scope.Lookup(ident.Name); obj != nil {
|
||||||
|
ident.Obj = obj
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// NewPackage uses an Importer to resolve imports. Given an importPath,
|
||||||
|
// an importer returns the imported package's name, its scope of exported
|
||||||
|
// objects, and an error, if any.
|
||||||
|
//
|
||||||
|
type Importer func(path string) (name string, scope *Scope, err os.Error)
|
||||||
|
|
||||||
|
|
||||||
|
// NewPackage creates a new Package node from a set of File nodes. It resolves
|
||||||
|
// unresolved identifiers across files and updates each file's Unresolved list
|
||||||
|
// accordingly. If a non-nil importer and universe scope are provided, they are
|
||||||
|
// used to resolve identifiers not declared in any of the package files. Any
|
||||||
|
// remaining unresolved identifiers are reported as undeclared. If the files
|
||||||
|
// belong to different packages, one package name is selected and files with
|
||||||
|
// different package name are reported and then ignored.
|
||||||
|
// The result is a package node and a scanner.ErrorList if there were errors.
|
||||||
|
//
|
||||||
|
func NewPackage(fset *token.FileSet, files map[string]*File, importer Importer, universe *Scope) (*Package, os.Error) {
|
||||||
|
var p pkgBuilder
|
||||||
|
p.fset = fset
|
||||||
|
|
||||||
|
// complete package scope
|
||||||
|
pkgName := ""
|
||||||
|
pkgScope := NewScope(universe)
|
||||||
|
for _, file := range files {
|
||||||
|
// package names must match
|
||||||
|
switch name := file.Name.Name; {
|
||||||
|
case pkgName == "":
|
||||||
|
pkgName = name
|
||||||
|
case name != pkgName:
|
||||||
|
p.errorf(file.Package, "package %s; expected %s", name, pkgName)
|
||||||
|
continue // ignore this file
|
||||||
|
}
|
||||||
|
|
||||||
|
// collect top-level file objects in package scope
|
||||||
|
for _, obj := range file.Scope.Objects {
|
||||||
|
p.declare(pkgScope, nil, obj)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// imports maps import paths to package names and scopes
|
||||||
|
// TODO(gri): Eventually we like to get to the import scope from
|
||||||
|
// a package object. Then we can have a map path -> Obj.
|
||||||
|
type importedPkg struct {
|
||||||
|
name string
|
||||||
|
scope *Scope
|
||||||
|
}
|
||||||
|
imports := make(map[string]*importedPkg)
|
||||||
|
|
||||||
|
// complete file scopes with imports and resolve identifiers
|
||||||
|
for _, file := range files {
|
||||||
|
// ignore file if it belongs to a different package
|
||||||
|
// (error has already been reported)
|
||||||
|
if file.Name.Name != pkgName {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// build file scope by processing all imports
|
||||||
|
importErrors := false
|
||||||
|
fileScope := NewScope(pkgScope)
|
||||||
|
for _, spec := range file.Imports {
|
||||||
|
// add import to global map of imports
|
||||||
|
path := string(spec.Path.Value)
|
||||||
|
path = path[1 : len(path)-1] // strip ""'s
|
||||||
|
pkg := imports[path]
|
||||||
|
if pkg == nil {
|
||||||
|
if importer == nil {
|
||||||
|
importErrors = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
name, scope, err := importer(path)
|
||||||
|
if err != nil {
|
||||||
|
p.errorf(spec.Path.Pos(), "could not import %s (%s)", path, err)
|
||||||
|
importErrors = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
pkg = &importedPkg{name, scope}
|
||||||
|
imports[path] = pkg
|
||||||
|
// TODO(gri) If a local package name != "." is provided,
|
||||||
|
// global identifier resolution could proceed even if the
|
||||||
|
// import failed. Consider adjusting the logic here a bit.
|
||||||
|
}
|
||||||
|
// local name overrides imported package name
|
||||||
|
name := pkg.name
|
||||||
|
if spec.Name != nil {
|
||||||
|
name = spec.Name.Name
|
||||||
|
}
|
||||||
|
// add import to file scope
|
||||||
|
if name == "." {
|
||||||
|
// merge imported scope with file scope
|
||||||
|
for _, obj := range pkg.scope.Objects {
|
||||||
|
p.declare(fileScope, pkgScope, obj)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// declare imported package object in file scope
|
||||||
|
obj := NewObj(Pkg, name)
|
||||||
|
obj.Decl = spec
|
||||||
|
p.declare(fileScope, pkgScope, obj)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// resolve identifiers
|
||||||
|
if importErrors {
|
||||||
|
// don't use the universe scope without correct imports
|
||||||
|
// (objects in the universe may be shadowed by imports;
|
||||||
|
// with missing imports identifiers might get resolved
|
||||||
|
// wrongly)
|
||||||
|
pkgScope.Outer = nil
|
||||||
|
}
|
||||||
|
i := 0
|
||||||
|
for _, ident := range file.Unresolved {
|
||||||
|
if !resolve(fileScope, ident) {
|
||||||
|
p.errorf(ident.Pos(), "undeclared name: %s", ident.Name)
|
||||||
|
file.Unresolved[i] = ident
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
file.Unresolved = file.Unresolved[0:i]
|
||||||
|
pkgScope.Outer = universe // reset universe scope
|
||||||
|
}
|
||||||
|
|
||||||
|
// collect all import paths and respective package scopes
|
||||||
|
importedScopes := make(map[string]*Scope)
|
||||||
|
for path, pkg := range imports {
|
||||||
|
importedScopes[path] = pkg.scope
|
||||||
|
}
|
||||||
|
|
||||||
|
return &Package{pkgName, pkgScope, importedScopes, files}, p.GetError(scanner.Sorted)
|
||||||
|
}
|
@ -39,16 +39,14 @@ func (s *Scope) Lookup(name string) *Object {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Insert attempts to insert a named object into the scope s.
|
// Insert attempts to insert a named object obj into the scope s.
|
||||||
// If the scope does not contain an object with that name yet,
|
// If the scope already contains an object alt with the same name,
|
||||||
// Insert inserts the object and returns it. Otherwise, Insert
|
// Insert leaves the scope unchanged and returns alt. Otherwise
|
||||||
// leaves the scope unchanged and returns the object found in
|
// it inserts obj and returns nil."
|
||||||
// the scope instead.
|
|
||||||
//
|
//
|
||||||
func (s *Scope) Insert(obj *Object) (alt *Object) {
|
func (s *Scope) Insert(obj *Object) (alt *Object) {
|
||||||
if alt = s.Objects[obj.Name]; alt == nil {
|
if alt = s.Objects[obj.Name]; alt == nil {
|
||||||
s.Objects[obj.Name] = obj
|
s.Objects[obj.Name] = obj
|
||||||
alt = obj
|
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -101,6 +99,11 @@ func (obj *Object) Pos() token.Pos {
|
|||||||
return n.Pos()
|
return n.Pos()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
case *ImportSpec:
|
||||||
|
if d.Name != nil && d.Name.Name == name {
|
||||||
|
return d.Name.Pos()
|
||||||
|
}
|
||||||
|
return d.Path.Pos()
|
||||||
case *ValueSpec:
|
case *ValueSpec:
|
||||||
for _, n := range d.Names {
|
for _, n := range d.Names {
|
||||||
if n.Name == name {
|
if n.Name == name {
|
||||||
|
@ -159,7 +159,8 @@ func ParseFiles(fset *token.FileSet, filenames []string, mode uint) (pkgs map[st
|
|||||||
name := src.Name.Name
|
name := src.Name.Name
|
||||||
pkg, found := pkgs[name]
|
pkg, found := pkgs[name]
|
||||||
if !found {
|
if !found {
|
||||||
pkg = &ast.Package{name, nil, make(map[string]*ast.File)}
|
// TODO(gri) Use NewPackage here; reconsider ParseFiles API.
|
||||||
|
pkg = &ast.Package{name, nil, nil, make(map[string]*ast.File)}
|
||||||
pkgs[name] = pkg
|
pkgs[name] = pkg
|
||||||
}
|
}
|
||||||
pkg.Files[filename] = src
|
pkg.Files[filename] = src
|
||||||
|
@ -148,8 +148,7 @@ func (p *parser) declare(decl interface{}, scope *ast.Scope, kind ast.ObjKind, i
|
|||||||
// remember the corresponding declaration for redeclaration
|
// remember the corresponding declaration for redeclaration
|
||||||
// errors and global variable resolution/typechecking phase
|
// errors and global variable resolution/typechecking phase
|
||||||
obj.Decl = decl
|
obj.Decl = decl
|
||||||
alt := scope.Insert(obj)
|
if alt := scope.Insert(obj); alt != nil && p.mode&DeclarationErrors != 0 {
|
||||||
if alt != obj && p.mode&DeclarationErrors != 0 {
|
|
||||||
prevDecl := ""
|
prevDecl := ""
|
||||||
if pos := alt.Pos(); pos.IsValid() {
|
if pos := alt.Pos(); pos.IsValid() {
|
||||||
prevDecl = fmt.Sprintf("\n\tprevious declaration at %s", p.file.Position(pos))
|
prevDecl = fmt.Sprintf("\n\tprevious declaration at %s", p.file.Position(pos))
|
||||||
@ -175,8 +174,9 @@ func (p *parser) shortVarDecl(idents []*ast.Ident) {
|
|||||||
// and are not global => no need to remember the respective
|
// and are not global => no need to remember the respective
|
||||||
// declaration
|
// declaration
|
||||||
alt := p.topScope.Insert(obj)
|
alt := p.topScope.Insert(obj)
|
||||||
if alt == obj {
|
if alt == nil {
|
||||||
n++ // new declaration
|
n++ // new declaration
|
||||||
|
alt = obj
|
||||||
}
|
}
|
||||||
ident.Obj = alt
|
ident.Obj = alt
|
||||||
}
|
}
|
||||||
@ -1151,12 +1151,16 @@ func (p *parser) parseElement(keyOk bool) ast.Expr {
|
|||||||
return p.parseLiteralValue(nil)
|
return p.parseLiteralValue(nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
x := p.parseRhs()
|
x := p.parseExpr(keyOk) // don't resolve if map key
|
||||||
if keyOk && p.tok == token.COLON {
|
if keyOk {
|
||||||
|
if p.tok == token.COLON {
|
||||||
colon := p.pos
|
colon := p.pos
|
||||||
p.next()
|
p.next()
|
||||||
x = &ast.KeyValueExpr{x, colon, p.parseElement(false)}
|
return &ast.KeyValueExpr{x, colon, p.parseElement(false)}
|
||||||
}
|
}
|
||||||
|
p.resolve(x) // not a map key
|
||||||
|
}
|
||||||
|
|
||||||
return x
|
return x
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2247,5 +2251,5 @@ func (p *parser) parseFile() *ast.File {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// TODO(gri): store p.imports in AST
|
// TODO(gri): store p.imports in AST
|
||||||
return &ast.File{doc, pos, ident, decls, p.pkgScope, p.unresolved[0:i], p.comments}
|
return &ast.File{doc, pos, ident, decls, p.pkgScope, p.imports, p.unresolved[0:i], p.comments}
|
||||||
}
|
}
|
||||||
|
@ -33,7 +33,7 @@ func (tc *typechecker) declInScope(scope *ast.Scope, kind ast.ObjKind, name *ast
|
|||||||
//obj.N = n
|
//obj.N = n
|
||||||
name.Obj = obj
|
name.Obj = obj
|
||||||
if name.Name != "_" {
|
if name.Name != "_" {
|
||||||
if alt := scope.Insert(obj); alt != obj {
|
if alt := scope.Insert(obj); alt != nil {
|
||||||
tc.Errorf(name.Pos(), "%s already declared at %s", name.Name, tc.fset.Position(alt.Pos()).String())
|
tc.Errorf(name.Pos(), "%s already declared at %s", name.Name, tc.fset.Position(alt.Pos()).String())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -53,7 +53,7 @@ func CheckPackage(fset *token.FileSet, pkg *ast.Package, importer Importer) os.E
|
|||||||
//
|
//
|
||||||
func CheckFile(fset *token.FileSet, file *ast.File, importer Importer) os.Error {
|
func CheckFile(fset *token.FileSet, file *ast.File, importer Importer) os.Error {
|
||||||
// create a single-file dummy package
|
// create a single-file dummy package
|
||||||
pkg := &ast.Package{file.Name.Name, nil, map[string]*ast.File{fset.Position(file.Name.NamePos).Filename: file}}
|
pkg := &ast.Package{file.Name.Name, nil, nil, map[string]*ast.File{fset.Position(file.Name.NamePos).Filename: file}}
|
||||||
return CheckPackage(fset, pkg, importer)
|
return CheckPackage(fset, pkg, importer)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -327,8 +327,8 @@ func (tc *typechecker) checkBlock(body []ast.Stmt, ftype *Type) {
|
|||||||
if ftype != nil {
|
if ftype != nil {
|
||||||
for _, par := range ftype.Params.Objects {
|
for _, par := range ftype.Params.Objects {
|
||||||
if par.Name != "_" {
|
if par.Name != "_" {
|
||||||
obj := tc.topScope.Insert(par)
|
alt := tc.topScope.Insert(par)
|
||||||
assert(obj == par) // ftype has no double declarations
|
assert(alt == nil) // ftype has no double declarations
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ var Universe *ast.Scope
|
|||||||
|
|
||||||
func def(obj *ast.Object) {
|
func def(obj *ast.Object) {
|
||||||
alt := Universe.Insert(obj)
|
alt := Universe.Insert(obj)
|
||||||
if alt != obj {
|
if alt != nil {
|
||||||
panic("object declared twice")
|
panic("object declared twice")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user