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

cmd/compile,wiki,spec: clarify whether or not the panics created in loop bodies should propagate to function iterators #71830

Open
zigo101 opened this issue Feb 19, 2025 · 9 comments
Assignees
Labels
compiler/runtime Issues related to the Go compiler and/or runtime. Documentation Issues describing a change to documentation. NeedsInvestigation Someone must examine and confirm this is a valid issue and not a duplicate of an existing one.
Milestone

Comments

@zigo101
Copy link

zigo101 commented Feb 19, 2025

By the current doc, the loop body is viewed as being directly nested both in the innermost nesting function of the loop and in the yield function which is called in the iterator function.

In my opinion, the loop body should be only viewed as directly nested both in the innermost nesting function of the loop.
In other words, when a function is used as an iterator, the yield calls should viewed as being removed from the iterator function.
The effect of the doc change is that panics created in loop body are not catch-able in the iterator function.

(Another explanation: the panics created in loop bodies don't propagate to iterator functions. The yield call just returns a false when panicking,)

For example, the following two programs should have the same behavior, but they don't in 1.24.0 implementation. This is weird to users. By adopting the above explanations, then their behavior will become the same (the two recover calls should be both no-op).

package main

import "fmt"

func main() {
	defer foo()
}

func foo() {
	for range iter {}
	panic(123)
}

func iter(yield func(int) bool) {
	defer func() {
		fmt.Println(recover())
	}()
	yield(0)
}
package main

import "fmt"

func main() {
	defer foo()
}

func foo() {
	for range iter {
		panic(123)
	}
}

func iter(yield func(int) bool) {
	defer func() {
		fmt.Println(recover())
	}()
	yield(0)
}
@gopherbot gopherbot added the Documentation Issues describing a change to documentation. label Feb 19, 2025
@zigo101
Copy link
Author

zigo101 commented Feb 19, 2025

[edit]: sorry, pls ignored this comment. It looks I over-thought about it. It already works as such now. So the problem is much simpler than I thought.

@zigo101 zigo101 changed the title doc: in ranging over an iterator function, clarify the relation of the loop body and the call to the iterator function spec and doc: in ranging over an iterator function, clarify the relation of the loop body and the call to the iterator function Feb 19, 2025
@zigo101 zigo101 changed the title spec and doc: in ranging over an iterator function, clarify the relation of the loop body and the call to the iterator function spec,doc: in ranging over an iterator function, clarify the relation of the loop body and the call to the iterator function Feb 19, 2025
@seankhliao seankhliao changed the title spec,doc: in ranging over an iterator function, clarify the relation of the loop body and the call to the iterator function spec: clarify scopes for function iterators Feb 19, 2025
@seankhliao
Copy link
Member

cc @rsc

@mknyszek
Copy link
Contributor

CC @dr2chase

@mknyszek mknyszek added the NeedsInvestigation Someone must examine and confirm this is a valid issue and not a duplicate of an existing one. label Feb 19, 2025
@mknyszek mknyszek added this to the Backlog milestone Feb 19, 2025
@ianlancetaylor ianlancetaylor added the compiler/runtime Issues related to the Go compiler and/or runtime. label Feb 19, 2025
@zigo101 zigo101 changed the title spec: clarify scopes for function iterators cmd/compile,wiki: clarify whether or not the panics created in loop bodies should propagate to function iterators Feb 19, 2025
@zigo101 zigo101 changed the title cmd/compile,wiki: clarify whether or not the panics created in loop bodies should propagate to function iterators cmd/compile,wiki, spec: clarify whether or not the panics created in loop bodies should propagate to function iterators Feb 23, 2025
@zigo101 zigo101 changed the title cmd/compile,wiki, spec: clarify whether or not the panics created in loop bodies should propagate to function iterators cmd/compile,wiki,spec: clarify whether or not the panics created in loop bodies should propagate to function iterators Feb 23, 2025
@zigo101
Copy link
Author

zigo101 commented Mar 13, 2025

ping.

IMO, this is a fundamental problem which should be resolved as early as possible,

@zigo101
Copy link
Author

zigo101 commented Mar 14, 2025

The following example is some ridiculous, but the crash message is absolutely inaccurate:

package main

import "fmt"
import "time"

func main() {
	for v := range abc {
		time.Sleep(5 * time.Second)
		fmt.Println(v)
	}
}

func abc(yield func(string) bool) {
	defer fmt.Println("bye")
	go func() {
		_ = yield("a")
	}()
	time.Sleep(time.Second)
}

Output:

bye
panic: runtime error: range function recovered a loop body panic and did not resume panicking

goroutine 1 [running]:
...

[edit]: maybe a new issue should be opened for this: the yield function should be only called in the goroutine executing the corresponding iterator.

A time.Sleep free example:

package main

import "fmt"

var c1 = make(chan int)
var c2 = make(chan int)

func main() {
	for range abc {
		c1 <- 1
		<-c2
	}
	fmt.Println("c")
	c2 <- 1
	fmt.Println("d")
}

func abc(yield func() bool) {
	defer func() {
		fmt.Println("recover: ", recover())
	}()
	fmt.Println("a")
	go func() {
		_ = yield()
	}()
	<-c1
	fmt.Println("b")
}

@zigo101
Copy link
Author

zigo101 commented Mar 15, 2025

The change actually breaks backward compatibility. However, the bad effect of the current design is larger than the breaking.

@zigo101
Copy link
Author

zigo101 commented Mar 17, 2025

Panics can propagate to different goroutines now. Intended?

package main

import "fmt"
import "time"

func main() {
	for range abc {
		panic(123)
	}
}

func abc(yield func() bool) {
	for range time.Tick(time.Second) {
		go func() {
			defer func() {
				fmt.Println("recover in go:", recover())
			}()
			_ = yield()
		}()
	}
}

If a yield returns false on panicking, then this will not happen.

@zigo101
Copy link
Author

zigo101 commented Mar 17, 2025

The following program prints 789 now. I am not sure whether this is correct or not if yield returns false on panicking.

package main

import "fmt"

func main() {
	for v := range abc {
		defer func() {
			fmt.Println(recover())
		}()
		fmt.Println(v)
		panic(789)
	}
}

func abc(yield func(int) bool) {
	defer yield(1)
	panic(123)
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
compiler/runtime Issues related to the Go compiler and/or runtime. Documentation Issues describing a change to documentation. NeedsInvestigation Someone must examine and confirm this is a valid issue and not a duplicate of an existing one.
Projects
Development

No branches or pull requests

7 participants