Source file 
src/runtime/unsafepoint_test.go
     1  
     2  
     3  
     4  
     5  package runtime_test
     6  
     7  import (
     8  	"internal/testenv"
     9  	"os"
    10  	"os/exec"
    11  	"reflect"
    12  	"runtime"
    13  	"strconv"
    14  	"strings"
    15  	"testing"
    16  )
    17  
    18  
    19  
    20  func setGlobalPointer() {
    21  	globalPointer = nil
    22  }
    23  
    24  var globalPointer *int
    25  
    26  func TestUnsafePoint(t *testing.T) {
    27  	testenv.MustHaveExec(t)
    28  	switch runtime.GOARCH {
    29  	case "amd64", "arm64":
    30  	default:
    31  		t.Skipf("test not enabled for %s", runtime.GOARCH)
    32  	}
    33  
    34  	
    35  	
    36  	f := runtime.FuncForPC(reflect.ValueOf(setGlobalPointer).Pointer())
    37  
    38  	
    39  	
    40  	
    41  	
    42  	
    43  	cmd := exec.Command(testenv.GoToolPath(t), "tool", "objdump", "-s", "setGlobalPointer", os.Args[0])
    44  	out, err := cmd.CombinedOutput()
    45  	if err != nil {
    46  		t.Fatalf("can't objdump %v:\n%s", err, out)
    47  	}
    48  	lines := strings.Split(string(out), "\n")[1:]
    49  
    50  	
    51  	var entry uint64
    52  	var startedWB bool
    53  	var doneWB bool
    54  	instructionCount := 0
    55  	unsafeCount := 0
    56  	for _, line := range lines {
    57  		line = strings.TrimSpace(line)
    58  		t.Logf("%s", line)
    59  		parts := strings.Fields(line)
    60  		if len(parts) < 4 {
    61  			continue
    62  		}
    63  		if !strings.HasPrefix(parts[0], "unsafepoint_test.go:") {
    64  			continue
    65  		}
    66  		pc, err := strconv.ParseUint(parts[1][2:], 16, 64)
    67  		if err != nil {
    68  			t.Fatalf("can't parse pc %s: %v", parts[1], err)
    69  		}
    70  		if entry == 0 {
    71  			entry = pc
    72  		}
    73  		
    74  		
    75  		
    76  		unsafe := runtime.UnsafePoint(f.Entry() + uintptr(pc-entry))
    77  		t.Logf("unsafe: %v\n", unsafe)
    78  		instructionCount++
    79  		if unsafe {
    80  			unsafeCount++
    81  		}
    82  
    83  		
    84  		if startedWB && !doneWB && !unsafe {
    85  			t.Errorf("instruction %s must be marked unsafe, but isn't", parts[1])
    86  		}
    87  
    88  		
    89  		switch runtime.GOARCH {
    90  		case "arm64":
    91  			if parts[3] == "MOVWU" {
    92  				
    93  				
    94  				startedWB = true
    95  			}
    96  			if parts[3] == "MOVD" && parts[4] == "ZR," {
    97  				
    98  				
    99  				doneWB = true
   100  			}
   101  		case "amd64":
   102  			if parts[3] == "CMPL" {
   103  				startedWB = true
   104  			}
   105  			if parts[3] == "MOVQ" && parts[4] == "$0x0," {
   106  				doneWB = true
   107  			}
   108  		}
   109  	}
   110  
   111  	if instructionCount == 0 {
   112  		t.Errorf("no instructions")
   113  	}
   114  	if unsafeCount == instructionCount {
   115  		t.Errorf("no interruptible instructions")
   116  	}
   117  	
   118  	
   119  	
   120  	
   121  	
   122  }
   123  
View as plain text