-
Notifications
You must be signed in to change notification settings - Fork 17.9k
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, types2: 'any' in go/types.Universe should be of type Alias #66921
Comments
Note: this poses a rather unfortunate problem for switching GODEBUG values inside tests. |
Change https://go.dev/cl/580355 mentions this issue: |
@griesemer @adonovan after staring at the implications of this for a minute, I think the only way to do it properly is to insist that the only way to change @timothy-king has created a |
Why does it matter exactly? The defined types
What would be the fix? What would the
Arguably this is evidence that it's not the right fix.
We should flip that builder to "0", since we don't need a "1" builder: it's the default at tip. |
The spec says that they are aliases, and there are various places where we need to check for distinguished pointers to print the right name. (search for types2?.Universe.Lookup("any") to find these special cases). Several of those places also have apologies that the "hack" wouldn't be necessary with proper alias support. Now we have that support. It is inconsistent for them not to be Aliases. Why shouldn't we fix this properly?
A new TypeName? I don't understand the question.
I don't see how that follows. We have a global that uses the new API. Any time a global is affected by a GODEBUG value, it seems that such a fix would be required. I don't understand the philosophical point. |
Sorry, I was under the misapprehension that all TypeNames had valid positions, but of course that's not true for all the built-in types. Ignore me. |
Thanks. Will wait for @griesemer to opine before spending too much more time here. See the associated CL for the proposed fix. It wasn't hard to get go/types/types2 tests passing, and I suspect just a few more changes are required in the compiler / importers. Do we want to do this? It's now or never, I suppose. Aside:
FWIW, I think that's no problem. Basic types are defined types. But Put differently, don't we eventually want the definition of |
Yeah, it's definitely a simpler representation. The question is how much churn will it generate. Only experience will tell. |
Hmm, well |
I'm worried about introducing more problems with this than solving. |
To be clearer about this: we could define |
@griesemer we already have special handling for Leaving |
@findleyr Sounds good. So maybe |
@griesemer that would be problematic for tools that still use gotypesalias=0. I think we need to change the representation based on the GODEBUG environment variable at startup. Perhaps we can just get away with inconsistent behavior if GODEBUG is subsequently modified in tests... This may also be a problem for bootstrapping. |
"Once at startup" seems like the only logical way to choose the representation of a global variable, but many of our tests need to call Setenv to 1 or 0 later on, and this may lead to inconsistency. Perhaps we need to establish and assume that all our tests are run in both modes, so tests need only skip when run with the opposite mode from what they need. |
Perhaps it would be less trouble to make this change later, once we've ditched gotypesalias=0 forever. Then there is no mode-dependence; we've just made the representation of |
@adonovan I think we should rip the band-aid off now. The sooner we make this change, the sooner we can avoid workarounds like https://go-review.git.corp.google.com/c/tools/+/580835/4/internal/refactor/inline/inline.go#2874. |
That workaround seems a lot simpler to me than the problem of initializing a global based on the value of an environment variable that we expect tests to modify. |
@adonovan that is just one of several, and it's likely there are more workarounds for any that don't exist yet should. I don't see why we should wait, and then have to make yet another breaking change in Go 1.24 or Go 1.25. |
My gut feeling is that the less ergonomic testing could be a major hardship for users of go/types, whereas the specific representation of the type If I understand correctly, the workaround in the above code is just to detect shadowing declarations of |
Decision: we're going to switch the implementation of I'm doing this in the above CL (as well as adding some safeguards against improper concurrent use), but as always it's proving trickier than one would hope. Should be able to get it working tomorrow morning. |
Change https://go.dev/cl/582335 mentions this issue: |
The isEmptyInterface predicate had a TODO noting that it should probably be considering the underlying type. After looking at usage, I agree that this was probably simple any oversight, but didn't matter because most uses of the empty interface were uses of any or interface{}, both of which were an interface. But with CL 580355, as the 'any' type is becoming a types.Alias, and this logic is caused inference scoring to change and gopls tests to fail. Fix isEmptyInterface to check underlying, and add a bit of commentary for the future. For golang/go#66921 Change-Id: I7ba3db1b04d83dda0372d9c39b943965f4d8c955 Reviewed-on: https://go-review.googlesource.com/c/tools/+/582335 Reviewed-by: Alan Donovan <adonovan@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
When GODEBUG=gotypesalias=1 is set, use an actual Alias type to represent any, rather than a legacy alias representation. This makes any consistent with other interface aliases, and will eventually make obsolete the various workarounds for formatting any as 'any' rather than 'interface{}'. Since any is a global in the Universe scope, we must hijack Scope.Lookup to select the correct representation. Of course, this also means that we can't support type checking concurrently while mutating gotypesalias (or, in the case of types2, Config.EnableAlias). Some care is taken to ensure that the type checker panics in the event of this type of misuse. For now, we must still support the legacy representation of any, and the existing workarounds that look for a distinguished any pointer. This is done by ensuring that both representations have the same underlying pointer, and by updating workarounds to consider Underlying. Fixes golang/go#66921 Change-Id: I81db7e8e15317b7a6ed3b406545db15a2fc42f57 Reviewed-on: https://go-review.googlesource.com/c/go/+/580355 Reviewed-by: Alan Donovan <adonovan@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
Just noticed that the predeclared types
any
,inbyte
, andrune
go/types.Universe
are not actually of typego/types.Alias
. This matters: I noticed it while working on type writing for the x/tools inliner.I'll put together a fix. I would not be surprised if additional tests fail as a result of this fix, since this would be the first time we have an
Alias
type in the universe scope.(EDIT: per discussion, let's just fix this for
any
. Basic types are special anyway).CC @griesemer @adonovan
The text was updated successfully, but these errors were encountered: