Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

go/types: API allows instantiation of instantiated type #72978

Open
griesemer opened this issue Mar 20, 2025 · 3 comments
Open

go/types: API allows instantiation of instantiated type #72978

griesemer opened this issue Mar 20, 2025 · 3 comments
Labels
BugReport Issues describing a possible bug in the Go implementation. NeedsFix The path to resolution is known, but the work has not been done.
Milestone

Comments

@griesemer
Copy link
Contributor

griesemer commented Mar 20, 2025

Consider: https://go.dev/play/p/SiV_jaPuzNr
Re-Instantiation of the instantiated type of x in the code works unexpectedly.
Re-Instantiation of the instantiated type of f in the code fails as expected.

This seems inconsistent. Either we permit both or neither.

@findleyr for input

@griesemer griesemer added the BugReport Issues describing a possible bug in the Go implementation. label Mar 20, 2025
@griesemer griesemer added this to the Go1.25 milestone Mar 20, 2025
@findleyr
Copy link
Member

This is just a bug: there are no type params to substitute in the double-instantiated instance. The underlying is the same as the first instance.
https://go.dev/play/p/6WDvAzt01tY

It seems we are just missing a check that there are no type arguments.

@cherrymui cherrymui added the NeedsFix The path to resolution is known, but the work has not been done. label Mar 21, 2025
@KirtanSoni
Copy link

May I propose a PR or a Plan on this?

@obeyda
Copy link
Contributor

obeyda commented Mar 27, 2025

Hi,
It looks like the double instantiation check was removed in CL 333589 when the instantiation logic was moved from subst.go to instantiate.go.

Current state:

switch orig := orig.(type) {
case *Named:
res = check.newNamedInstance(pos, orig, targs, expanding) // substituted lazily

I suggest to fix this by reintroducing the check in instantiate.go. For example:

 case *Named:
+    // Verify type parameter count for named types
+    tparams := orig.TypeParams()
+    if !check.validateTArgLen(pos, orig.obj.Name(), tparams.Len(), len(targs)) {
+        return Typ[Invalid]
+    }
+    if tparams.Len() == 0 {
+        // No type parameters to substitute (orig is not generic or already instantiated)
+        return orig  // nothing to do
+    }
     res = check.newNamedInstance(pos, orig, targs, expanding) // substituted lazily

This should restore the original behavior and prevent double instantiation. Happy to work on a CL if this approach looks acceptable.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
BugReport Issues describing a possible bug in the Go implementation. NeedsFix The path to resolution is known, but the work has not been done.
Projects
None yet
Development

No branches or pull requests

5 participants