| // Copyright 2023 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. | |
| //go:build !plan9 && !windows | |
| // Test that callbacks from C to Go in the same C-thread always get the same m. | |
| // Make sure the extra M bind to the C-thread. | |
| package main | |
| /* | |
| extern void CheckBindM(); | |
| */ | |
| import "C" | |
| import ( | |
| "fmt" | |
| "os" | |
| "runtime" | |
| "sync" | |
| "sync/atomic" | |
| ) | |
| var ( | |
| mutex = sync.Mutex{} | |
| cThreadToM = map[uintptr]uintptr{} | |
| started = atomic.Uint32{} | |
| ) | |
| // same as CTHREADS in C, make sure all the C threads are actually started. | |
| const cThreadNum = 2 | |
| func init() { | |
| register("EnsureBindM", EnsureBindM) | |
| } | |
| //export GoCheckBindM | |
| func GoCheckBindM(thread uintptr) { | |
| // Wait all threads start | |
| if started.Load() != cThreadNum { | |
| // Only once for each thread, since it will wait all threads start. | |
| started.Add(1) | |
| for started.Load() < cThreadNum { | |
| runtime.Gosched() | |
| } | |
| } | |
| m := runtime_getm_for_test() | |
| mutex.Lock() | |
| defer mutex.Unlock() | |
| if savedM, ok := cThreadToM[thread]; ok && savedM != m { | |
| fmt.Printf("m == %x want %x\n", m, savedM) | |
| os.Exit(1) | |
| } | |
| cThreadToM[thread] = m | |
| } | |
| func EnsureBindM() { | |
| C.CheckBindM() | |
| fmt.Println("OK") | |
| } | |