mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
Now that we have specific types for ONAME and ODCLFUNC nodes (*Name and *Func), use them throughout the compiler to be more precise about what data is being operated on. This is a somewhat large CL, but once you start applying the types in a few places, you end up needing to apply them to many other places to keep everything type-checking. A lot of code also melts away as types are added. Passes buildall w/ toolstash -cmp. Change-Id: I21dd9b945d701c470332bac5394fca744a5b232d Reviewed-on: https://go-review.googlesource.com/c/go/+/274097 Trust: Russ Cox <rsc@golang.org> Run-TryBot: Russ Cox <rsc@golang.org> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Matthew Dempsky <mdempsky@google.com>
235 lines
5.6 KiB
Go
235 lines
5.6 KiB
Go
// Copyright 2009 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.
|
|
|
|
package gc
|
|
|
|
import (
|
|
"cmd/compile/internal/base"
|
|
"cmd/compile/internal/ir"
|
|
"cmd/compile/internal/types"
|
|
"cmd/internal/bio"
|
|
"cmd/internal/src"
|
|
"fmt"
|
|
"go/constant"
|
|
)
|
|
|
|
func exportf(bout *bio.Writer, format string, args ...interface{}) {
|
|
fmt.Fprintf(bout, format, args...)
|
|
if base.Debug.Export != 0 {
|
|
fmt.Printf(format, args...)
|
|
}
|
|
}
|
|
|
|
var asmlist []ir.Node
|
|
|
|
// exportsym marks n for export (or reexport).
|
|
func exportsym(n ir.Node) {
|
|
if n.Sym().OnExportList() {
|
|
return
|
|
}
|
|
n.Sym().SetOnExportList(true)
|
|
|
|
if base.Flag.E != 0 {
|
|
fmt.Printf("export symbol %v\n", n.Sym())
|
|
}
|
|
|
|
exportlist = append(exportlist, n)
|
|
}
|
|
|
|
func initname(s string) bool {
|
|
return s == "init"
|
|
}
|
|
|
|
func autoexport(n ir.Node, ctxt ir.Class) {
|
|
if n.Sym().Pkg != ir.LocalPkg {
|
|
return
|
|
}
|
|
if (ctxt != ir.PEXTERN && ctxt != ir.PFUNC) || dclcontext != ir.PEXTERN {
|
|
return
|
|
}
|
|
if n.Type() != nil && n.Type().IsKind(types.TFUNC) && ir.IsMethod(n) {
|
|
return
|
|
}
|
|
|
|
if types.IsExported(n.Sym().Name) || initname(n.Sym().Name) {
|
|
exportsym(n)
|
|
}
|
|
if base.Flag.AsmHdr != "" && !n.Sym().Asm() {
|
|
n.Sym().SetAsm(true)
|
|
asmlist = append(asmlist, n)
|
|
}
|
|
}
|
|
|
|
func dumpexport(bout *bio.Writer) {
|
|
// The linker also looks for the $$ marker - use char after $$ to distinguish format.
|
|
exportf(bout, "\n$$B\n") // indicate binary export format
|
|
off := bout.Offset()
|
|
iexport(bout.Writer)
|
|
size := bout.Offset() - off
|
|
exportf(bout, "\n$$\n")
|
|
|
|
if base.Debug.Export != 0 {
|
|
fmt.Printf("BenchmarkExportSize:%s 1 %d bytes\n", base.Ctxt.Pkgpath, size)
|
|
}
|
|
}
|
|
|
|
func importsym(ipkg *types.Pkg, s *types.Sym, op ir.Op) ir.Node {
|
|
n := ir.AsNode(s.PkgDef())
|
|
if n == nil {
|
|
// iimport should have created a stub ONONAME
|
|
// declaration for all imported symbols. The exception
|
|
// is declarations for Runtimepkg, which are populated
|
|
// by loadsys instead.
|
|
if s.Pkg != Runtimepkg {
|
|
base.Fatalf("missing ONONAME for %v\n", s)
|
|
}
|
|
|
|
n = dclname(s)
|
|
s.SetPkgDef(n)
|
|
s.Importdef = ipkg
|
|
}
|
|
if n.Op() != ir.ONONAME && n.Op() != op {
|
|
redeclare(base.Pos, s, fmt.Sprintf("during import %q", ipkg.Path))
|
|
}
|
|
return n
|
|
}
|
|
|
|
// importtype returns the named type declared by symbol s.
|
|
// If no such type has been declared yet, a forward declaration is returned.
|
|
// ipkg is the package being imported
|
|
func importtype(ipkg *types.Pkg, pos src.XPos, s *types.Sym) *types.Type {
|
|
n := importsym(ipkg, s, ir.OTYPE)
|
|
if n.Op() != ir.OTYPE {
|
|
t := types.New(types.TFORW)
|
|
t.Sym = s
|
|
t.Nod = n
|
|
|
|
n.SetOp(ir.OTYPE)
|
|
n.SetPos(pos)
|
|
n.SetType(t)
|
|
n.SetClass(ir.PEXTERN)
|
|
}
|
|
|
|
t := n.Type()
|
|
if t == nil {
|
|
base.Fatalf("importtype %v", s)
|
|
}
|
|
return t
|
|
}
|
|
|
|
// importobj declares symbol s as an imported object representable by op.
|
|
// ipkg is the package being imported
|
|
func importobj(ipkg *types.Pkg, pos src.XPos, s *types.Sym, op ir.Op, ctxt ir.Class, t *types.Type) ir.Node {
|
|
n := importsym(ipkg, s, op)
|
|
if n.Op() != ir.ONONAME {
|
|
if n.Op() == op && (n.Class() != ctxt || !types.Identical(n.Type(), t)) {
|
|
redeclare(base.Pos, s, fmt.Sprintf("during import %q", ipkg.Path))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
n.SetOp(op)
|
|
n.SetPos(pos)
|
|
n.SetClass(ctxt)
|
|
if ctxt == ir.PFUNC {
|
|
n.Sym().SetFunc(true)
|
|
}
|
|
n.SetType(t)
|
|
return n
|
|
}
|
|
|
|
// importconst declares symbol s as an imported constant with type t and value val.
|
|
// ipkg is the package being imported
|
|
func importconst(ipkg *types.Pkg, pos src.XPos, s *types.Sym, t *types.Type, val constant.Value) {
|
|
n := importobj(ipkg, pos, s, ir.OLITERAL, ir.PEXTERN, t)
|
|
if n == nil { // TODO: Check that value matches.
|
|
return
|
|
}
|
|
|
|
n.SetVal(val)
|
|
|
|
if base.Flag.E != 0 {
|
|
fmt.Printf("import const %v %L = %v\n", s, t, val)
|
|
}
|
|
}
|
|
|
|
// importfunc declares symbol s as an imported function with type t.
|
|
// ipkg is the package being imported
|
|
func importfunc(ipkg *types.Pkg, pos src.XPos, s *types.Sym, t *types.Type) {
|
|
n := importobj(ipkg, pos, s, ir.ONAME, ir.PFUNC, t)
|
|
if n == nil {
|
|
return
|
|
}
|
|
name := n.(*ir.Name)
|
|
|
|
fn := ir.NewFunc(pos)
|
|
fn.SetType(t)
|
|
name.SetFunc(fn)
|
|
fn.Nname = name
|
|
|
|
if base.Flag.E != 0 {
|
|
fmt.Printf("import func %v%S\n", s, t)
|
|
}
|
|
}
|
|
|
|
// importvar declares symbol s as an imported variable with type t.
|
|
// ipkg is the package being imported
|
|
func importvar(ipkg *types.Pkg, pos src.XPos, s *types.Sym, t *types.Type) {
|
|
n := importobj(ipkg, pos, s, ir.ONAME, ir.PEXTERN, t)
|
|
if n == nil {
|
|
return
|
|
}
|
|
|
|
if base.Flag.E != 0 {
|
|
fmt.Printf("import var %v %L\n", s, t)
|
|
}
|
|
}
|
|
|
|
// importalias declares symbol s as an imported type alias with type t.
|
|
// ipkg is the package being imported
|
|
func importalias(ipkg *types.Pkg, pos src.XPos, s *types.Sym, t *types.Type) {
|
|
n := importobj(ipkg, pos, s, ir.OTYPE, ir.PEXTERN, t)
|
|
if n == nil {
|
|
return
|
|
}
|
|
|
|
if base.Flag.E != 0 {
|
|
fmt.Printf("import type %v = %L\n", s, t)
|
|
}
|
|
}
|
|
|
|
func dumpasmhdr() {
|
|
b, err := bio.Create(base.Flag.AsmHdr)
|
|
if err != nil {
|
|
base.Fatalf("%v", err)
|
|
}
|
|
fmt.Fprintf(b, "// generated by compile -asmhdr from package %s\n\n", ir.LocalPkg.Name)
|
|
for _, n := range asmlist {
|
|
if n.Sym().IsBlank() {
|
|
continue
|
|
}
|
|
switch n.Op() {
|
|
case ir.OLITERAL:
|
|
t := n.Val().Kind()
|
|
if t == constant.Float || t == constant.Complex {
|
|
break
|
|
}
|
|
fmt.Fprintf(b, "#define const_%s %#v\n", n.Sym().Name, n.Val())
|
|
|
|
case ir.OTYPE:
|
|
t := n.Type()
|
|
if !t.IsStruct() || t.StructType().Map != nil || t.IsFuncArgStruct() {
|
|
break
|
|
}
|
|
fmt.Fprintf(b, "#define %s__size %d\n", n.Sym().Name, int(t.Width))
|
|
for _, f := range t.Fields().Slice() {
|
|
if !f.Sym.IsBlank() {
|
|
fmt.Fprintf(b, "#define %s_%s %d\n", n.Sym().Name, f.Sym.Name, int(f.Offset))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
b.Close()
|
|
}
|