golang / go

The Go programming language
https://go.dev
BSD 3-Clause "New" or "Revised" License
122.68k stars 17.49k forks source link

all: merge dev.typeparams to master during Go 1.17 #43931

Closed mdempsky closed 3 years ago

mdempsky commented 3 years ago

What

Proposal #43651 is to add generics to the Go language. The proposal currently has 1223 likes and only 89 dislikes. Demand for generics was a top requested feature in the 2016, 2017, 2018, and 2019 Go developer survey results. There's a working go2go Playground for trying out the proposal. The type parameters proposal was the subject of a keynote talk at GopherCon 2020. This is a widely demanded feature from the Go community; it's not a niche feature like persistent memory or field tracking.

This proposal then is about allowing prototype compiler development of that feature to happen on the master branch. The new language functionality would be kept behind a compiler flag until both (1) #43651 is accepted and (2) the functionality is ready for end users.

Of course, if #43651 is accepted before this proposal, then this one becomes moot. Alternatively, should #43651 be rejected after this one is accepted, then the implementation support code should be backed out of master.

Rationale

We've already begun development for #43651 on the dev.typeparams branch, but keeping it in sync with master and dev.regabi has been very tedious. Within the last 2 months of development, we've had 20+ merges between branches. We had a failed merge that required resetting the development branches (#43147). We've had compiler engineers attempt and give up merges, because the conflicts were too difficult to resolve without more familiarity/time.

This tedium will likely decrease somewhat after Go 1.16 is released and dev.regabi merges back into master, but then there will also be changes happening elsewhere in the repo to keep up with (e.g., #42637 caused a lot of trybot flakes on both dev.regabi and dev.typeparams). Also, support for generics will continue requiring more pervasive changes that involve touching the backend. There are also other accepted Go language changes slated for Go 1.17 (e.g., #395, #19367, #38248, #40481) that will overlap in frontend areas.

It would ease cmd/compile development significantly if the compiler functionality in dev.typeparams could be merged into master and developed behind a feature flag. Feature flags are how we developed binary export (ae2f54a7718e803fe00ee0e5b83e6c07036ba93f), SSA (c0740fed377109526e61955af612efba8ec3f3dc), package syntax's new parser (2ff463948c92e71651a31c621826281035a7071a), indexed export (ca2f85fd3f32b2a4c863a2de602876bd31e9d956), new escape analysis (97c4ad432743d74ee59648dee0db1b107c701834), etc. (I'm sure there are others; these are just ones I'm personally familiar with.)

Additionally, dev.typeparams introduces a new go/types-based typechecker "types2". Even if we reject #43651, I think we want to adopt this new typechecker to replace the compiler's legacy typechecker ("typecheck"). And because generic support in types2 is already conditional behind a flag, removing generics support if #43651 is rejected would be no different than when we removed the old code any of the above features replaced.

Flag

As for how to spell the compiler flag, dev.typeparams currently uses -G, but other spellings would be fine. The particular values assigned to it though are somewhat arbitrary, driven by development needs that no longer apply, so they should be revisited. I suggest -G=0 means use typecheck; -G=1 means use types2 w/o generics support; and -G=2 means use types2 w/ generics support. (I'm hopeful -G=1 can be the default for Go 1.17; then we can drop support for -G=0 for Go 1.18, which should make it easier to enable -G=2 by default. However, I'm not proposing either of these default changes at this time.)

It's been brought up that users might then use it and things could break. We currently provide users with compiler flags like -B (disable bounds checking), -wb=false (disable write barriers), and -d=disablenil (disable nil checking), so users have ample opportunities for self-harm today already.

But moreover, I think we want users trying out features before they're fully ready. It's been helpful having users report regressions on dev.regabi (#43479, #43480, #43701, #43818), and at the end of release cycles we're always begging for users to try the betas and release candidates.

If really desired, we could put it behind GOEXPERIMENT=typeparams with other features like field tracking and static lock ranking, which never went through the proposal process. But then I'd ask again for consideration of #42681.

mdempsky commented 3 years ago

This is a widely demanded feature from the Go community; it's not a niche feature like persistent memory or field tracking.

Just to clarify, this was not intended to deride #43810. Persistent memory was brought up as a counter-point in the Google compiler/runtime team meeting, which I understood at the time to be a hypothetical proposal. I vaguely remembered the topic of Go language support for persistent memory being discussed in the past over email, but I was not aware it had yet advanced to a proposal.

I'm not presently familiar enough with persistent memory to know whether #43810 is a good idea, and I don't mean to imply either way about it here. But I do think its current thumbs up/down ratio demonstrates a different level of broad community support than #43651, which was the point I was arguing for why I think it's justified as in the interest of the broad Go community to allow development for #43651 to continue on master behind a feature flag, when we probably wouldn't extend the same offer to #43810.

rsc commented 3 years ago

I don't think we can land code on master before a proposal is accepted. That goes against the spirit of the proposal process and the rules we ask others to follow for proposed changes. I think the right way to view this proposal is as simply to start working on master once #43651 is accepted, changing our previous plan to stay on dev.typeparams until Go 1.18.

My bigger concern is how to properly firewall off the generics work from the Go 1.17 work so that changes being made to help set up for generics don't cause undue instability in the Go 1.17 release.

How many code changes do you contemplate (for example in the back ends) that would not be gated by the flag?

komuw commented 3 years ago

That goes against the spirit of the proposal process...

English isn't my first language, but that line sounds circular to me.

No proposal can be against the spirit of the proposal process, it is a proposal.

mvdan commented 3 years ago

No proposal can be against the spirit of the proposal process, it is a proposal.

I think Russ is being taken slightly out of context there. I understood his comment to say: we've historically not merged code before a proposal is accepted, so we should not make an exception here. It might set a bad precedent, which is something I can understand.

That said, I think this proposal is special, given how much it's refactoring and improving the compiler and typechecker. I assume we want those merged in the future, even if generics end up being rejected. So I understand Matthew's point about the large amount of extra tedious work involved, which could be avoided.

andybons-stripe commented 3 years ago

Landing code on master implementing a proposal before it is accepted is at odds with what the proposal process exists for. There is no implication that the filing of this proposal goes against that. I can perhaps see how the confusion arose, so please consider this to be a clarifying point, as I was in attendance when @rsc’s response was written.

DeedleFake commented 3 years ago

No proposal can be against the spirit of the proposal process, it is a proposal.

That's not what he's saying. He's saying that the point of the proposal process is to have a standardized way to get new features into the language. If a proposal, in this case #43651, is given special treatment and allowed to circumvent part of that process, then it is unfair to other proposals and sets a bad precedent for later ones because people will want to know why their popular but still unaccepted proposal can't get that treatment, too.

hherman1 commented 3 years ago

Few proposals have the scope of the generics work. I think it’s ok to make exceptions, and if people need to understand why exceptions were made it’s easy to explain.

komuw commented 3 years ago

If a proposal,..., is given special treatment and allowed to circumvent ...

Not all proposals will have the same scope. For some proposals, no design document is required. For others, a design document is required. Other proposals come with a proto-type implementation(go-modules as an example).

I think what is important is for an explanation to be given as to why a certain proposal requires an exemption to the rules. For you to obtain such an exemption, what do you do? You write a proposal. That is what Mdempsky has done.

It is up to others, then to figure out if the reasons given are enough to warrant the exemption.

bcmills commented 3 years ago

To my knowledge #43651 still doesn't even have a specific proposal for the changes to the language spec (see https://github.com/golang/go/issues/43651#issuecomment-758854504), which seems like a necessary step for any serious proposal to change the language. But perhaps that step is waiting on detail to be informed by the implementation work.

43651 sketches out a design that requires a lot of refactoring and implementation work. Some of the refactoring work seems outside the scope of the proposal process in the first place (since it has no user-facing effect on the toolchain). On top of that, the user-facing implementation work seems like a fine application for GOEXPERIMENT, which could be used to provide test coverage.

I would much rather we accept the implementation work as a GOEXPERIMENT than prematurely accept the language change (without the actual text of the language change!) in order to unblock implementation work. We shouldn't have to pass the bill to find out what's in it.

bcmills commented 3 years ago

My bigger concern is how to properly firewall off the generics work from the Go 1.17 work so that changes being made to help set up for generics don't cause undue instability in the Go 1.17 release.

As the de facto primary build-cop for the project over the past couple of years, I'm honestly more concerned with baking in regressions on the branch.

We already have what is in my opinion a severe problem with regressions being introduced into the tree and dismissed as “flakes”, or accepted as likely bugs but not triaged promptly because they occur with low frequency. With regressions introduced on a separate branch, it is even easier to (intentionally or accidentally) bury regressions that happen to manifest as intermittent failures.

I would rather we deal with those regressions in the main branch when they are introduced, rather than accumulating a backlog of regressions in the branch and spending a month or more dealing with an unstable tree after the branch is merged back in.

ainar-g commented 3 years ago

I really can't comment on the “spirit of the proposal process” line, but isn't the goal of the proposal process making sure that the changes are well designed? Features often get designed one way only for that original design to be improved in the process of implementing it and discovering its inconsistencies and omissions. The GO111MODULE experiment is one example. I don't have the precise data, but something tells me that Go modules also have been a feature touching a lot of code, and it has been developed (and improved) on the main branch for six releases.

Regarding the possible instability in Go 1.17. Type parameters are possibly the most anticipated feature since Go has been publicly released. I think it's reasonable to assume that many people will build Go from the main branch once the type parameters work is merged. They will want to try the new features, play with them, and report bugs. The bugs some of which will show flaws in the design. Especially if there is a call for help to the community, for example through the Go Blog.

rogpeppe commented 3 years ago

Would it be possible to accept #43651 in principle without requiring the proposal to be finalised to the very last detail?

ISTM that the basic outline of the proposal is solid, and there's considerable room for movement around remaining semantic details while still keeping the overall model intact. Once #43651 is accepted in principle, further proposals could be introduced to nail down specific semantics for some of the more awkward parts, but ISTM that those are unlikely to invalidate the preparatory work currently being undertaken in the dev.typeparams branch.

That would allow work to start on the master branch even while details are being finalised.

bcmills commented 3 years ago

Would it be possible to accept #43651 in principle without requiring the proposal to be finalised to the very last detail?

Maybe? But that's in tension with Russ's comment:

That goes against the spirit of the proposal process and the rules we ask others to follow for proposed changes.

“[T]he rules we ask others to follow for proposed changes” are, in part, answering the questions in the Go 2 language change template, including:

  • Please describe as precisely as possible the change to the language.
  • What would change in the language spec?
  • Please also describe the change informally, as in a class teaching Go. …
    • How would the language spec change?
    • Orthogonality: how does this change interact or overlap with existing features?

I believe that the current design document satisfies only the first of those five points. In my opinion, “a class teaching Go” would need to include answers to basic questions like “what is a type?” and “what is an interface?”, which are covered in the existing spec but notably missing from the current proposal — despite the concerns I raised last summer and re-raised in https://github.com/golang/go/issues/43651#issuecomment-758854504 about those concepts.

rsc commented 3 years ago

This has gotten a bit heated, and even though it has cooled down, I'm going to lock it until next week.

rsc commented 3 years ago

Last week I wrote:

I think the right way to view this proposal is as simply to start working on master once #43651 is accepted, changing our previous plan to stay on dev.typeparams until Go 1.18.

Continuing to view this proposal as "merge dev.typeparams to master during Go 1.17 dev cycle", this seems like a reasonable thing to do. I also asked:

My bigger concern is how to properly firewall off the generics work from the Go 1.17 work so that changes being made to help set up for generics don't cause undue instability in the Go 1.17 release.

How many code changes do you contemplate (for example in the back ends) that would not be gated by the flag?

We should make sure that we are careful to keep risky changes gated by the flag.

We should also probably disable the flag in the Go 1.17 release itself, so that we don't end up with an ecosystem of packages that have to be built with -gcflags=all=-generics.

Are there other precautions we should take to make sure that merging this code to master does not cause undue risk in Go 1.17?

rsc commented 3 years ago

I should add that one of the most compelling reasons for making this change is that it creates backpressure on non-generics work to make sure that the work is compatible with what's going on in dev.typeparams. Without the merge to master, all the merges from master end up creating work for the team working on generics, with no feedback to the master-branch work that changes are being made incompatibly.

beoran commented 3 years ago

I'm just as eager as anyone for #43651 to be accepted, however I agree with @rsc and @bcmills that we should follow the procedure for language changes. I know I did when I made my failed proposals >;-> So the question is then: how can we speed up the acceptance of #43651? Could the Go team shift a few people, or perhaps ask Google if they are willing to hire some more help? Can we, outside of the Go team, contribute somehow, perhaps with the informal description if the feature?

Once #43651 is accepted, this proposal could also be accepted to speed up implementation then, hidden behind a flag or GOEXPERIMENT, whichever is most effective and convenient.

mdempsky commented 3 years ago

How many code changes do you contemplate (for example in the back ends) that would not be gated by the flag?

I see code changes falling into three camps:

  1. Generics-specific code that will eventually replace existing code paths, which can be gated behind a flag. For example, the new parser/typechecker/noder code on dev.typeparams.
  2. Generics-specific code that is purely additive. For example, I expect additional IR types will be necessary to represent generic functions, to represent dictionaries, etc. These are harder to gate behind a flag, but I expect would just lie unused when generics aren't enabled. If necessary, we can add runtime assertions to make sure they're only used when generics are enabled.
  3. General code refactorings to facilitate the above changes. Stuff like all the package ir refactoring that landed on dev.regabi (but unlikely anything further to that same scale).

I think the camp 1 changes are largely done already on dev.typeparams; I think we've already run that option about as far as possible. I think most remaining work will be split between camps 2 and 3. I don't currently have any solid predictions of what the ratio might be, but I expect there will be significant work still falling into each camp and diffuse through a lot of parts of the compiler.

For example, I think we'll want DWARF support for generic code/types. This will probably require both refactoring some of the current DWARF code (camp 3), and then adding support for generating the new DWARF DIEs (camp 2). The refactored code will continue benefiting from existing tests and development practices (e.g., testing CLs with toolstash -cmp), while the new code should have new tests but also naturally won't be used in Go 1.17 with generics disabled anyway.

I'd generally expect changes to be more in camp 2 closer to the frontend (e.g., transforming generic code into IR), and more in camp 3 closer to the backend (e.g., plumbing generics metadata through as needed for debugging/runtime).

We should make sure that we are careful to keep risky changes gated by the flag.

Ack.

We should also probably disable the flag in the Go 1.17 release itself, so that we don't end up with an ecosystem of packages that have to be built with -gcflags=all=-generics.

I think disabling generics support in the Go 1.17 release builds would be fine.

More broadly, I think it would be nice to better delineate between compiler flags/extensions that are stable and supported for direct use by end users, and unstable flags that are only there for use by compiler developers and/or the standard library. But that's a separate issue.

Are there other precautions we should take to make sure that merging this code to master does not cause undue risk in Go 1.17?

I think the best precaution is merging as early as possible / reasonable. The master branch is the most tested/used development branch. The earlier dev.typeparams is merged, the more time+testing we'll have to make sure changes haven't negatively impacted the non-generics code paths.

Without the merge to master, all the merges from master end up creating work for the team working on generics, with no feedback to the master-branch work that changes are being made incompatibly.

Also, merges are already difficult enough to create and review, and there's no good workflow for breaking tricky merge conflicts into smaller CLs like you would with normal development, except to merge more frequently. Each merge is all or nothing.

Reviewing merge conflicts are also error-prone / tedious: Gerrit only presents the specific conflicts and how they were resolved. It doesn't provide an easy way to see the changes in their full, original contexts, so reviewers have to go out of their way to track down this information if needed.

Finally, mistakes introduced by bad merge conflict resolutions will be harder to diagnose later. E.g., if git bisect identifies a merge as the problematic CL, the human developer still needs to spend further time figuring out whether the merge conflicts were the source of trouble, or that functionality that was developed on the separate branches simply don't interoperate.

Overall, I think merging dev.typeparams to master for Go 1.17 and doing development on master does increase risk somewhat by simple nature of more changes -> more opportunity for mistakes. But I think that risk is already offset by the increased testing exposure working on master and also the additional engineer time freed up by not having to worry about merges. And I think it will also significantly reduce the risks for enabling generics in Go 1.18 (assuming #43651 is accepted).

mmaedel commented 3 years ago

When the master 'hijack' is taking place I'd like to have pipelined improvements from all relevant branches in the EXPERIMENT... so that we could see their effects on existing codebases in one Go2 pre-Release 'testing suite'.

rsc commented 3 years ago

Based on the discussion above, this proposal seems like a likely accept. — rsc for the proposal review group

rsc commented 3 years ago

No change in consensus, so accepted. 🎉 This issue now tracks the work of implementing the proposal. — rsc for the proposal review group

griesemer commented 3 years ago

https://go-review.googlesource.com/c/go/+/294331 has happened.

OneOfOne commented 3 years ago

O M G, I DID NOT THINK THIS WILL HAPPEN IN MY LIFE TIME.

elichai commented 3 years ago

How can I use this to play with generics? I'd like to compare some future serialization code using generics and how do generics and inline affect each other (obviously not building anything that I expect to keep working in the future, just want to play with the syntax and/or performance of the result)

Tried using the -G=2 but I get weird errors:

$ gotip run -gcflags=all=-G=2 main.go 
go build internal/unsafeheader: open /tmp/go-build1866398939/b005/_pkg_.a: no such file or directory
go build internal/race: open /tmp/go-build1866398939/b021/_pkg_.a: no such file or directory
go build internal/abi: open /tmp/go-build1866398939/b008/_pkg_.a: no such file or directory
go build math/bits: open /tmp/go-build1866398939/b017/_pkg_.a: no such file or directory
go build runtime/internal/sys: open /tmp/go-build1866398939/b013/_pkg_.a: no such file or directory
go build unicode/utf8: open /tmp/go-build1866398939/b019/_pkg_.a: no such file or directory
os.Stat of archive file failed: stat /tmp/go-build1866398939/b011/_pkg_.a: no such file or directory
mdempsky commented 3 years ago

@elichai Currently you still need to use -G=3. We haven't yet renumbered the flag values like I suggested in the proposal.

Beware that the generics-specific support is still very early. Non-generic Go code is expected to fully work, so please file issues if you find anything that works without -G=3 but breaks with -G=3. But don't be surprised if code using new generics features doesn't work yet.

gopherbot commented 3 years ago

Change https://golang.org/cl/295029 mentions this issue: cmd/compile: renumber -G flag values

dzpt commented 3 years ago

@mdempsky Hi, do you understand why -G=3 doesn't work with go module? if i remove go.mod file, it works, with it existance i got generics syntax error syntax error: unexpected [, expecting (

cuonglm commented 3 years ago

@mdempsky Hi, do you understand why -G=3 doesn't work with go module?

if i remove go.mod file, it works, with it existance i got generics syntax error syntax error: unexpected [, expecting (

What's exactly command do you run? Maybe try -gcflags=all=-G=3

dzpt commented 3 years ago

@cuonglm Ive tried with gotip run -gcflags=-G=3 . and gotip run -gcflags=all=-G=3

cuonglm commented 3 years ago

@dzpt then you may want to try dev.typeparams instead, not all generic features are available on tip.

dzpt commented 3 years ago

@cuonglm it works without go.mod, so i don't think generic code causes the error.

gotip run -gcflags=all=-G=3 .                   
# generics/xys
xys/xys.go:10:10: syntax error: unexpected [, expecting (
xys/xys.go:19:13: syntax error: unexpected [, expecting (
xys/xys.go:20:13: syntax error: unexpected ] after top level declaration

generics works on main.go, but got error on generics/xys.go

cuonglm commented 3 years ago

What version specify in your go.mod?

dzpt commented 3 years ago

@cuonglm 1.17

module generics

go 1.17
cuonglm commented 3 years ago

@cuonglm 1.17


module generics

go 1.17

Then I dont think it will work, as generic syntax is only available in go1.18

dzpt commented 3 years ago

@cuonglm then you might not read this issue 1.17 has early generics feature. like i said, it WORKS without go mod

https://github.com/mattn/go-generics-example checkout this repo.

cuonglm commented 3 years ago

@dzpt Can you share your code instead? (Sorry for confusion in previous comment, what I mean is not all generic syntax is available in go1.17)

Also, can be helpful if you can post go run -x or go build -x when using go modules.

dzpt commented 3 years ago

@cuonglm the code is here https://github.com/dzpt/test-go-generics uncomment Find method in xys/xys.go and main.go line 19,20

it works fine with same generics code in main.go, but failed for xys/xys.go

cuonglm commented 3 years ago

@dzpt I got this:

⚡ cat main.go xys/xys.go
package main

import (
    "fmt"
    "generics/xys"
)

func mapFunc[T any, M any](a []T, f func(T) M) []M {
    n := make([]M, len(a), cap(a))
    for i, e := range a {
        n[i] = f(e)
    }
    return n
}

// gotip run -gcflags=-G=3 main.go
func main() {
    a := []int{1, 2, 3}
    xys.Find(1, a)
    xys.Abc()
    vi := []int{1, 2, 3, 4, 5, 6}
    vs := mapFunc(vi, func(v int) string {
        return "<" + fmt.Sprint(v) + ">"
    })
    fmt.Println(vs)
}
package xys

import "fmt"

func Abc() {
    fmt.Println("123")
}

// func Find[T any, []T any](what T, where []T) (idx int) {
func Find[T comparable](what T, where []T) int {
    for i, v := range where {
        if v == what {
            return i
        }
    }
    return -1
}

// func mapFunc[T any, M any](a []T, f func(T) M) []M {
//  n := make([]M, len(a), cap(a))
//  for i, e := range a {
//      n[i] = f(e)
//  }
//  return n
// }

⚡ gotip run -gcflags=all=-G=3 main.go
# generics/xys
xys/xys.go:10:6: internal compiler error: Cannot export a generic function (yet): Find

goroutine 1 [running]:
runtime/debug.Stack()
    /Users/cuonglm/sdk/gotip/src/runtime/debug/stack.go:24 +0x88
cmd/compile/internal/base.FatalfAt({0x2, 0xa060}, {0x1012a80af, 0x2a}, {0x14000387398, 0x1, 0x1})
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/base/print.go:227 +0x1c0
cmd/compile/internal/noder.(*irgen).objFinish(0x1400017a870, 0x140003cbba0, 0x7, 0x1400037fa40)
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/object.go:168 +0x2ec
cmd/compile/internal/noder.(*irgen).objCommon(0x1400017a870, {0x2, 0xa060}, 0x1, 0x1400016d4f0, 0x7, 0x1400037fa40)
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/object.go:137 +0x68
cmd/compile/internal/noder.(*irgen).obj(0x1400017a870, {0x1014b7858, 0x1400012cae0})
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/object.go:99 +0x934
cmd/compile/internal/noder.(*irgen).def(0x1400017a870, 0x14000060600)
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/object.go:22 +0xf4
cmd/compile/internal/noder.(*irgen).funcDecl(0x1400017a870, 0x140003875f0, 0x14000063e60)
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/decl.go:85 +0xac
cmd/compile/internal/noder.(*irgen).decls(0x1400017a870, {0x1400002f810, 0x2, 0x3})
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/decl.go:28 +0x1ac
cmd/compile/internal/noder.(*irgen).generate(0x1400017a870, {0x1400000e168, 0x1, 0x1})
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/irgen.go:160 +0x4b4
cmd/compile/internal/noder.check2({0x1400000e168, 0x1, 0x1})
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/irgen.go:78 +0x558
cmd/compile/internal/noder.LoadPackage({0x1400000a340, 0x1, 0x2})
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/noder/noder.go:80 +0x328
cmd/compile/internal/gc.Main(0x1014965f0)
    /Users/cuonglm/sdk/gotip/src/cmd/compile/internal/gc/main.go:192 +0xcf4
main.main()
    /Users/cuonglm/sdk/gotip/src/cmd/compile/main.go:55 +0x144

Like I said, not all generic features available in go1.17 . I don't think go modules is relevant here. My gotip version:

⚡ gotip version
go version devel go1.17-0914646 Fri Jul 23 18:10:55 2021 +0000 darwin/arm64
mdempsky commented 3 years ago

@dzpt Sorry, generics are not supported in Go 1.17. Full stop.

If you want to help test generics for Go 1.18, you can use the dev.typeparams branch and file new issues at golang.org/issue/new. This issue is closed.

Thanks.

dzpt commented 3 years ago

@cuonglm my gotip version

gotip version
go version devel go1.18-9f928f9318e Sat Jul 24 01:54:15 2021 +0000 darwin/amd64

@mdempsky already go 1.18 (dev.typeparams branch) , generics works on main file but couldn't export from other package?

cuonglm commented 3 years ago

@dzpt please upgrade your module file to require 1.18, if you use above gotip version, then it will work.

~/test-go-generics [master●] » cat go.mod
module generics

go 1.18
~/test-go-generics [master●] » go run -gcflags=all=-G=3 main.go
123
[<1> <2> <3> <4> <5> <6>]
~/test-go-generics [master●] » go version
go version devel go1.18-9f928f9318 Sat Jul 24 01:54:15 2021 +0000 darwin/arm64
cuonglm commented 3 years ago

@dzpt Also note that your above one technically is not gotip. When you said gotip yesterday, I thought you use the official gotip, which will build whatever on master branch.

Your above comment indicate that you are using dev.typeparams, so the code structure, and support generic code are way difference compare to master. That why when your module require go1.17, your gotip can't parse generic syntax.

dzpt commented 3 years ago

@cuonglm thank you, it works now after changing go mod 1.17 to 1.18