mirror of
				https://github.com/golang/go.git
				synced 2025-10-26 14:24:14 +00:00 
			
		
		
		
	cmd/compile: mark concrete call of reflect.(*rtype).Method as REFLECTMETHOD
For functions that call reflect.Type.Method (or MethodByName), we mark it as REFLECTMETHOD, which tells the linker that methods can be retrieved via reflection and the linker keeps all exported methods live. Currently, this marking expects exactly the interface call reflect.Type.Method (or MethodByName). But now the compiler can devirtualize that call to a concrete call reflect.(*rtype).Method (or MethodByName), which is not handled and causing the linker to discard methods too aggressively. Handle the latter in this CL. Fixes #44207. Change-Id: Ia4060472dbff6ab6a83d2ca8e60a3e3f180ee832 Reviewed-on: https://go-review.googlesource.com/c/go/+/290950 Trust: Cherry Zhang <cherryyz@google.com> Reviewed-by: Matthew Dempsky <mdempsky@google.com>
This commit is contained in:
		
							parent
							
								
									e9c9683597
								
							
						
					
					
						commit
						ed8079096f
					
				
					 2 changed files with 39 additions and 1 deletions
				
			
		|  | @ -550,8 +550,12 @@ opswitch: | |||
| 	case OCLOSUREVAR, OCFUNC: | ||||
| 
 | ||||
| 	case OCALLINTER, OCALLFUNC, OCALLMETH: | ||||
| 		if n.Op == OCALLINTER { | ||||
| 		if n.Op == OCALLINTER || n.Op == OCALLMETH { | ||||
| 			// We expect both interface call reflect.Type.Method and concrete | ||||
| 			// call reflect.(*rtype).Method. | ||||
| 			usemethod(n) | ||||
| 		} | ||||
| 		if n.Op == OCALLINTER { | ||||
| 			markUsedIfaceMethod(n) | ||||
| 		} | ||||
| 
 | ||||
|  | @ -3710,6 +3714,16 @@ func usemethod(n *Node) { | |||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// Don't mark reflect.(*rtype).Method, etc. themselves in the reflect package. | ||||
| 	// Those functions may be alive via the itab, which should not cause all methods | ||||
| 	// alive. We only want to mark their callers. | ||||
| 	if myimportpath == "reflect" { | ||||
| 		switch Curfn.Func.Nname.Sym.Name { // TODO: is there a better way than hardcoding the names? | ||||
| 		case "(*rtype).Method", "(*rtype).MethodByName", "(*interfaceType).Method", "(*interfaceType).MethodByName": | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// Note: Don't rely on res0.Type.String() since its formatting depends on multiple factors | ||||
| 	//       (including global variables such as numImports - was issue #19028). | ||||
| 	// Also need to check for reflect package itself (see Issue #38515). | ||||
|  |  | |||
							
								
								
									
										24
									
								
								test/reflectmethod7.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								test/reflectmethod7.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,24 @@ | |||
| // run | ||||
| 
 | ||||
| // Copyright 2021 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
| 
 | ||||
| // See issue 44207. | ||||
| 
 | ||||
| package main | ||||
| 
 | ||||
| import "reflect" | ||||
| 
 | ||||
| type S int | ||||
| 
 | ||||
| func (s S) M() {} | ||||
| 
 | ||||
| func main() { | ||||
| 	t := reflect.TypeOf(S(0)) | ||||
| 	fn, ok := reflect.PtrTo(t).MethodByName("M") | ||||
| 	if !ok { | ||||
| 		panic("FAIL") | ||||
| 	} | ||||
| 	fn.Func.Call([]reflect.Value{reflect.New(t)}) | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Cherry Zhang
						Cherry Zhang