Skip to content

Commit 0a4fc72

Browse files
committed
internal/gcimporter: support materialized aliases
This CL adds support for preserving materialized aliases while types transit export data (indexed or unified). (This CL depends on CL 574737 to the compiler and go/types.) Updates golang/go#65294 Updates golang/go#64581 Change-Id: I1a0a08357e4f6a480ba6250fbea327922e455873 Reviewed-on: https://go-review.googlesource.com/c/tools/+/574717 LUCI-TryBot-Result: Go LUCI <[email protected]> Reviewed-by: Robert Findley <[email protected]>
1 parent 85b6527 commit 0a4fc72

File tree

6 files changed

+79
-26
lines changed

6 files changed

+79
-26
lines changed

internal/gcimporter/gcimporter_test.go

+1
Original file line numberDiff line numberDiff line change
@@ -190,6 +190,7 @@ func TestImportTypeparamTests(t *testing.T) {
190190
"nested.go": "fails to compile", // TODO(rfindley): investigate this.
191191
"issue47631.go": "can not handle local type declarations",
192192
"issue55101.go": "fails to compile",
193+
"issue50259.go": "compiler crashes if GODEBUG=gotypesalias=1", // TODO(adonovan): delete when #66550 is fixed.
193194
}
194195
}
195196

internal/gcimporter/iexport.go

+35-10
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import (
2121
"sort"
2222
"strconv"
2323
"strings"
24+
"unsafe"
2425

2526
"golang.org/x/tools/go/types/objectpath"
2627
"golang.org/x/tools/internal/aliases"
@@ -464,7 +465,7 @@ func (p *iexporter) doDecl(obj types.Object) {
464465

465466
switch obj := obj.(type) {
466467
case *types.Var:
467-
w.tag('V')
468+
w.tag(varTag)
468469
w.pos(obj.Pos())
469470
w.typ(obj.Type(), obj.Pkg())
470471

@@ -482,9 +483,9 @@ func (p *iexporter) doDecl(obj types.Object) {
482483

483484
// Function.
484485
if sig.TypeParams().Len() == 0 {
485-
w.tag('F')
486+
w.tag(funcTag)
486487
} else {
487-
w.tag('G')
488+
w.tag(genericFuncTag)
488489
}
489490
w.pos(obj.Pos())
490491
// The tparam list of the function type is the declaration of the type
@@ -500,15 +501,15 @@ func (p *iexporter) doDecl(obj types.Object) {
500501
w.signature(sig)
501502

502503
case *types.Const:
503-
w.tag('C')
504+
w.tag(constTag)
504505
w.pos(obj.Pos())
505506
w.value(obj.Type(), obj.Val())
506507

507508
case *types.TypeName:
508509
t := obj.Type()
509510

510511
if tparam, ok := aliases.Unalias(t).(*types.TypeParam); ok {
511-
w.tag('P')
512+
w.tag(typeParamTag)
512513
w.pos(obj.Pos())
513514
constraint := tparam.Constraint()
514515
if p.version >= iexportVersionGo1_18 {
@@ -523,8 +524,13 @@ func (p *iexporter) doDecl(obj types.Object) {
523524
}
524525

525526
if obj.IsAlias() {
526-
w.tag('A')
527+
w.tag(aliasTag)
527528
w.pos(obj.Pos())
529+
if alias, ok := t.(*aliases.Alias); ok {
530+
// Preserve materialized aliases,
531+
// even of non-exported types.
532+
t = aliasRHS(alias)
533+
}
528534
w.typ(t, obj.Pkg())
529535
break
530536
}
@@ -536,9 +542,9 @@ func (p *iexporter) doDecl(obj types.Object) {
536542
}
537543

538544
if named.TypeParams().Len() == 0 {
539-
w.tag('T')
545+
w.tag(typeTag)
540546
} else {
541-
w.tag('U')
547+
w.tag(genericTypeTag)
542548
}
543549
w.pos(obj.Pos())
544550

@@ -548,7 +554,7 @@ func (p *iexporter) doDecl(obj types.Object) {
548554
w.tparamList(obj.Name(), named.TypeParams(), obj.Pkg())
549555
}
550556

551-
underlying := obj.Type().Underlying()
557+
underlying := named.Underlying()
552558
w.typ(underlying, obj.Pkg())
553559

554560
if types.IsInterface(t) {
@@ -739,7 +745,10 @@ func (w *exportWriter) doTyp(t types.Type, pkg *types.Package) {
739745
}()
740746
}
741747
switch t := t.(type) {
742-
// TODO(adonovan): support types.Alias.
748+
case *aliases.Alias:
749+
// TODO(adonovan): support parameterized aliases, following *types.Named.
750+
w.startType(aliasType)
751+
w.qualifiedType(t.Obj())
743752

744753
case *types.Named:
745754
if targs := t.TypeArgs(); targs.Len() > 0 {
@@ -1322,3 +1331,19 @@ func (e internalError) Error() string { return "gcimporter: " + string(e) }
13221331
func internalErrorf(format string, args ...interface{}) error {
13231332
return internalError(fmt.Sprintf(format, args...))
13241333
}
1334+
1335+
// aliasRHS removes exactly one Alias constructor.
1336+
func aliasRHS(alias *aliases.Alias) types.Type {
1337+
// TODO(adonovan): if proposal #66559 is accepted, this will
1338+
// become Alias.RHS(alias). In the meantime, we must punch
1339+
// through the drywall.
1340+
type go123Alias struct {
1341+
_ *types.TypeName
1342+
_ *types.TypeParamList
1343+
RHS types.Type
1344+
_ types.Type
1345+
}
1346+
var raw *go123Alias
1347+
*(**aliases.Alias)(unsafe.Pointer(&raw)) = alias
1348+
return raw.RHS
1349+
}

internal/gcimporter/iexport_go118_test.go

+6
Original file line numberDiff line numberDiff line change
@@ -123,6 +123,12 @@ func TestImportTypeparamTests(t *testing.T) {
123123
t.Fatal(err)
124124
}
125125

126+
// TODO(adonovan): delete when #66550 is fixed.
127+
if strings.Contains(os.Getenv("GODEBUG"), "gotypesalias=1") &&
128+
entry.Name() == "issue50259.go" {
129+
t.Skip("Skipping test of defined<->alias cycle under gotypesaliases=1 (#66550)")
130+
}
131+
126132
if !bytes.HasPrefix(src, []byte("// run")) && !bytes.HasPrefix(src, []byte("// compile")) {
127133
// We're bypassing the logic of run.go here, so be conservative about
128134
// the files we consider in an attempt to make this test more robust to

internal/gcimporter/iexport_test.go

+5-2
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ import (
3232
"golang.org/x/tools/go/buildutil"
3333
"golang.org/x/tools/go/gcexportdata"
3434
"golang.org/x/tools/go/loader"
35+
"golang.org/x/tools/internal/aliases"
3536
"golang.org/x/tools/internal/gcimporter"
3637
"golang.org/x/tools/internal/testenv"
3738
"golang.org/x/tools/internal/typeparams/genericfeatures"
@@ -145,6 +146,7 @@ type UnknownType undefined
145146
t.Errorf("Loaded only %d packages, want at least %d", numPkgs, want)
146147
}
147148

149+
// TODO(adonovan): opt: parallelize this slow loop.
148150
for _, pkg := range sorted {
149151
if exportdata, err := iexport(conf.Fset, version, pkg); err != nil {
150152
t.Error(err)
@@ -333,14 +335,15 @@ func cmpObj(x, y types.Object) error {
333335
if xalias, yalias := x.IsAlias(), y.(*types.TypeName).IsAlias(); xalias != yalias {
334336
return fmt.Errorf("mismatching IsAlias(): %s vs %s", x, y)
335337
}
338+
336339
// equalType does not recurse into the underlying types of named types, so
337340
// we must pass the underlying type explicitly here. However, in doing this
338341
// we may skip checking the features of the named types themselves, in
339342
// situations where the type name is not referenced by the underlying or
340343
// any other top-level declarations. Therefore, we must explicitly compare
341344
// named types here, before passing their underlying types into equalType.
342-
xn, _ := xt.(*types.Named)
343-
yn, _ := yt.(*types.Named)
345+
xn, _ := aliases.Unalias(xt).(*types.Named)
346+
yn, _ := aliases.Unalias(yt).(*types.Named)
344347
if (xn == nil) != (yn == nil) {
345348
return fmt.Errorf("mismatching types: %T vs %T", xt, yt)
346349
}

internal/gcimporter/iimport.go

+31-13
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,20 @@ const (
8080
typeParamType
8181
instanceType
8282
unionType
83+
aliasType
84+
)
85+
86+
// Object tags
87+
const (
88+
varTag = 'V'
89+
funcTag = 'F'
90+
genericFuncTag = 'G'
91+
constTag = 'C'
92+
aliasTag = 'A'
93+
genericAliasTag = 'B'
94+
typeParamTag = 'P'
95+
typeTag = 'T'
96+
genericTypeTag = 'U'
8397
)
8498

8599
// IImportData imports a package from the serialized package data
@@ -324,7 +338,7 @@ func iimportCommon(fset *token.FileSet, getPackages GetPackagesFunc, data []byte
324338
}
325339

326340
// SetConstraint can't be called if the constraint type is not yet complete.
327-
// When type params are created in the 'P' case of (*importReader).obj(),
341+
// When type params are created in the typeParamTag case of (*importReader).obj(),
328342
// the associated constraint type may not be complete due to recursion.
329343
// Therefore, we defer calling SetConstraint there, and call it here instead
330344
// after all types are complete.
@@ -546,33 +560,37 @@ func (r *importReader) obj(name string) {
546560
pos := r.pos()
547561

548562
switch tag {
549-
case 'A':
563+
case aliasTag:
550564
typ := r.typ()
551-
552-
r.declare(types.NewTypeName(pos, r.currPkg, name, typ))
553-
554-
case 'C':
565+
// TODO(adonovan): support generic aliases:
566+
// if tag == genericAliasTag {
567+
// tparams := r.tparamList()
568+
// alias.SetTypeParams(tparams)
569+
// }
570+
r.declare(aliases.NewAlias(pos, r.currPkg, name, typ))
571+
572+
case constTag:
555573
typ, val := r.value()
556574

557575
r.declare(types.NewConst(pos, r.currPkg, name, typ, val))
558576

559-
case 'F', 'G':
577+
case funcTag, genericFuncTag:
560578
var tparams []*types.TypeParam
561-
if tag == 'G' {
579+
if tag == genericFuncTag {
562580
tparams = r.tparamList()
563581
}
564582
sig := r.signature(nil, nil, tparams)
565583
r.declare(types.NewFunc(pos, r.currPkg, name, sig))
566584

567-
case 'T', 'U':
585+
case typeTag, genericTypeTag:
568586
// Types can be recursive. We need to setup a stub
569587
// declaration before recursing.
570588
obj := types.NewTypeName(pos, r.currPkg, name, nil)
571589
named := types.NewNamed(obj, nil, nil)
572590
// Declare obj before calling r.tparamList, so the new type name is recognized
573591
// if used in the constraint of one of its own typeparams (see #48280).
574592
r.declare(obj)
575-
if tag == 'U' {
593+
if tag == genericTypeTag {
576594
tparams := r.tparamList()
577595
named.SetTypeParams(tparams)
578596
}
@@ -604,7 +622,7 @@ func (r *importReader) obj(name string) {
604622
}
605623
}
606624

607-
case 'P':
625+
case typeParamTag:
608626
// We need to "declare" a typeparam in order to have a name that
609627
// can be referenced recursively (if needed) in the type param's
610628
// bound.
@@ -637,7 +655,7 @@ func (r *importReader) obj(name string) {
637655
// completely set up all types in ImportData.
638656
r.p.later = append(r.p.later, setConstraintArgs{t: t, constraint: constraint})
639657

640-
case 'V':
658+
case varTag:
641659
typ := r.typ()
642660

643661
r.declare(types.NewVar(pos, r.currPkg, name, typ))
@@ -854,7 +872,7 @@ func (r *importReader) doType(base *types.Named) (res types.Type) {
854872
errorf("unexpected kind tag in %q: %v", r.p.ipath, k)
855873
return nil
856874

857-
case definedType:
875+
case aliasType, definedType:
858876
pkg, name := r.qualifiedIdent()
859877
r.p.doDecl(pkg, name)
860878
return pkg.Scope().Lookup(name).(*types.TypeName).Type()

internal/gcimporter/ureader_yes.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -524,7 +524,7 @@ func (pr *pkgReader) objIdx(idx pkgbits.Index) (*types.Package, string) {
524524
case pkgbits.ObjAlias:
525525
pos := r.pos()
526526
typ := r.typ()
527-
declare(types.NewTypeName(pos, objPkg, objName, typ))
527+
declare(aliases.NewAlias(pos, objPkg, objName, typ))
528528

529529
case pkgbits.ObjConst:
530530
pos := r.pos()

0 commit comments

Comments
 (0)