mirror of
https://github.com/golang/go
synced 2024-11-26 17:46:57 -07:00
cmd/compile: enable panic+recover adjustment for some ABI wrappers
For most ABI wrappers we don't need it because we're never going to defer an ABI wrapper for a function that then recovers, so that's would just be unnecessary code in the ABI wrapper. However, for functions that could be on the path of invoking a deferred function that can recover (runtime.reflectcall, reflect.callReflect, and reflect.callMethod), we do want the panic+recover adjustment. Set the Wrapper flag for them. Currently, those functions are defined as ABIInternal to avoid the ABI wrappers. But the assembly code still follows ABI0 calling convention, which would not work with the register-based calling convention. In particlar, it is not possible to make runtime.reflectcall ABIInternal, because it tail calls runtime.callNN functions, which are splittable. Later CLs will make them ABI0 and use the wrappers. Updates #40724. Change-Id: Ic7a45bbc6f726d29b5cb4932951a9d71578dcaf6 Reviewed-on: https://go-review.googlesource.com/c/go/+/307236 Trust: Cherry Zhang <cherryyz@google.com> Run-TryBot: Cherry Zhang <cherryyz@google.com> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Austin Clements <austin@google.com>
This commit is contained in:
parent
79b2e14b1a
commit
0723f062ff
@ -393,10 +393,20 @@ func setupTextLSym(f *ir.Func, flag int) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Clumsy but important.
|
// Clumsy but important.
|
||||||
|
// For functions that could be on the path of invoking a deferred
|
||||||
|
// function that can recover (runtime.reflectcall, reflect.callReflect,
|
||||||
|
// and reflect.callMethod), we want the panic+recover special handling.
|
||||||
// See test/recover.go for test cases and src/reflect/value.go
|
// See test/recover.go for test cases and src/reflect/value.go
|
||||||
// for the actual functions being considered.
|
// for the actual functions being considered.
|
||||||
if base.Ctxt.Pkgpath == "reflect" {
|
//
|
||||||
switch f.Sym().Name {
|
// runtime.reflectcall is an assembly function which tailcalls
|
||||||
|
// WRAPPER functions (runtime.callNN). Its ABI wrapper needs WRAPPER
|
||||||
|
// flag as well.
|
||||||
|
fnname := f.Sym().Name
|
||||||
|
if base.Ctxt.Pkgpath == "runtime" && fnname == "reflectcall" {
|
||||||
|
flag |= obj.WRAPPER
|
||||||
|
} else if base.Ctxt.Pkgpath == "reflect" {
|
||||||
|
switch fnname {
|
||||||
case "callReflect", "callMethod":
|
case "callReflect", "callMethod":
|
||||||
flag |= obj.WRAPPER
|
flag |= obj.WRAPPER
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user