mirror of
https://github.com/golang/go
synced 2024-11-20 06:44:40 -07:00
5288fadbdc
Add actual tracing of interesting runtime events. Part of a larger tracing functionality: https://docs.google.com/document/u/1/d/1FP5apqzBgr7ahCCgFO-yoVhk4YZrNIDNf9RybngBc14/pub Full change: https://codereview.appspot.com/146920043 Change-Id: Icccf54aea54e09350bb698ba6bf11532f9fbe6d3 Reviewed-on: https://go-review.googlesource.com/1451 Reviewed-by: Russ Cox <rsc@golang.org>
290 lines
6.2 KiB
Go
290 lines
6.2 KiB
Go
// Copyright 2009 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.
|
|
|
|
// Semaphore implementation exposed to Go.
|
|
// Intended use is provide a sleep and wakeup
|
|
// primitive that can be used in the contended case
|
|
// of other synchronization primitives.
|
|
// Thus it targets the same goal as Linux's futex,
|
|
// but it has much simpler semantics.
|
|
//
|
|
// That is, don't think of these as semaphores.
|
|
// Think of them as a way to implement sleep and wakeup
|
|
// such that every sleep is paired with a single wakeup,
|
|
// even if, due to races, the wakeup happens before the sleep.
|
|
//
|
|
// See Mullender and Cox, ``Semaphores in Plan 9,''
|
|
// http://swtch.com/semaphore.pdf
|
|
|
|
package runtime
|
|
|
|
import "unsafe"
|
|
|
|
// Asynchronous semaphore for sync.Mutex.
|
|
|
|
type semaRoot struct {
|
|
lock mutex
|
|
head *sudog
|
|
tail *sudog
|
|
nwait uint32 // Number of waiters. Read w/o the lock.
|
|
}
|
|
|
|
// Prime to not correlate with any user patterns.
|
|
const semTabSize = 251
|
|
|
|
var semtable [semTabSize]struct {
|
|
root semaRoot
|
|
pad [_CacheLineSize - unsafe.Sizeof(semaRoot{})]byte
|
|
}
|
|
|
|
//go:linkname sync_runtime_Semacquire sync.runtime_Semacquire
|
|
func sync_runtime_Semacquire(addr *uint32) {
|
|
semacquire(addr, true)
|
|
}
|
|
|
|
//go:linkname net_runtime_Semacquire net.runtime_Semacquire
|
|
func net_runtime_Semacquire(addr *uint32) {
|
|
semacquire(addr, true)
|
|
}
|
|
|
|
//go:linkname sync_runtime_Semrelease sync.runtime_Semrelease
|
|
func sync_runtime_Semrelease(addr *uint32) {
|
|
semrelease(addr)
|
|
}
|
|
|
|
//go:linkname net_runtime_Semrelease net.runtime_Semrelease
|
|
func net_runtime_Semrelease(addr *uint32) {
|
|
semrelease(addr)
|
|
}
|
|
|
|
// Called from runtime.
|
|
func semacquire(addr *uint32, profile bool) {
|
|
gp := getg()
|
|
if gp != gp.m.curg {
|
|
throw("semacquire not on the G stack")
|
|
}
|
|
|
|
// Easy case.
|
|
if cansemacquire(addr) {
|
|
return
|
|
}
|
|
|
|
// Harder case:
|
|
// increment waiter count
|
|
// try cansemacquire one more time, return if succeeded
|
|
// enqueue itself as a waiter
|
|
// sleep
|
|
// (waiter descriptor is dequeued by signaler)
|
|
s := acquireSudog()
|
|
root := semroot(addr)
|
|
t0 := int64(0)
|
|
s.releasetime = 0
|
|
if profile && blockprofilerate > 0 {
|
|
t0 = cputicks()
|
|
s.releasetime = -1
|
|
}
|
|
for {
|
|
lock(&root.lock)
|
|
// Add ourselves to nwait to disable "easy case" in semrelease.
|
|
xadd(&root.nwait, 1)
|
|
// Check cansemacquire to avoid missed wakeup.
|
|
if cansemacquire(addr) {
|
|
xadd(&root.nwait, -1)
|
|
unlock(&root.lock)
|
|
break
|
|
}
|
|
// Any semrelease after the cansemacquire knows we're waiting
|
|
// (we set nwait above), so go to sleep.
|
|
root.queue(addr, s)
|
|
goparkunlock(&root.lock, "semacquire", traceEvGoBlockSync)
|
|
if cansemacquire(addr) {
|
|
break
|
|
}
|
|
}
|
|
if s.releasetime > 0 {
|
|
blockevent(int64(s.releasetime)-t0, 3)
|
|
}
|
|
releaseSudog(s)
|
|
}
|
|
|
|
func semrelease(addr *uint32) {
|
|
root := semroot(addr)
|
|
xadd(addr, 1)
|
|
|
|
// Easy case: no waiters?
|
|
// This check must happen after the xadd, to avoid a missed wakeup
|
|
// (see loop in semacquire).
|
|
if atomicload(&root.nwait) == 0 {
|
|
return
|
|
}
|
|
|
|
// Harder case: search for a waiter and wake it.
|
|
lock(&root.lock)
|
|
if atomicload(&root.nwait) == 0 {
|
|
// The count is already consumed by another goroutine,
|
|
// so no need to wake up another goroutine.
|
|
unlock(&root.lock)
|
|
return
|
|
}
|
|
s := root.head
|
|
for ; s != nil; s = s.next {
|
|
if s.elem == unsafe.Pointer(addr) {
|
|
xadd(&root.nwait, -1)
|
|
root.dequeue(s)
|
|
break
|
|
}
|
|
}
|
|
unlock(&root.lock)
|
|
if s != nil {
|
|
if s.releasetime != 0 {
|
|
s.releasetime = cputicks()
|
|
}
|
|
goready(s.g)
|
|
}
|
|
}
|
|
|
|
func semroot(addr *uint32) *semaRoot {
|
|
return &semtable[(uintptr(unsafe.Pointer(addr))>>3)%semTabSize].root
|
|
}
|
|
|
|
func cansemacquire(addr *uint32) bool {
|
|
for {
|
|
v := atomicload(addr)
|
|
if v == 0 {
|
|
return false
|
|
}
|
|
if cas(addr, v, v-1) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
|
|
func (root *semaRoot) queue(addr *uint32, s *sudog) {
|
|
s.g = getg()
|
|
s.elem = unsafe.Pointer(addr)
|
|
s.next = nil
|
|
s.prev = root.tail
|
|
if root.tail != nil {
|
|
root.tail.next = s
|
|
} else {
|
|
root.head = s
|
|
}
|
|
root.tail = s
|
|
}
|
|
|
|
func (root *semaRoot) dequeue(s *sudog) {
|
|
if s.next != nil {
|
|
s.next.prev = s.prev
|
|
} else {
|
|
root.tail = s.prev
|
|
}
|
|
if s.prev != nil {
|
|
s.prev.next = s.next
|
|
} else {
|
|
root.head = s.next
|
|
}
|
|
s.elem = nil
|
|
s.next = nil
|
|
s.prev = nil
|
|
}
|
|
|
|
// Synchronous semaphore for sync.Cond.
|
|
type syncSema struct {
|
|
lock mutex
|
|
head *sudog
|
|
tail *sudog
|
|
}
|
|
|
|
// syncsemacquire waits for a pairing syncsemrelease on the same semaphore s.
|
|
//go:linkname syncsemacquire sync.runtime_Syncsemacquire
|
|
func syncsemacquire(s *syncSema) {
|
|
lock(&s.lock)
|
|
if s.head != nil && s.head.nrelease > 0 {
|
|
// Have pending release, consume it.
|
|
var wake *sudog
|
|
s.head.nrelease--
|
|
if s.head.nrelease == 0 {
|
|
wake = s.head
|
|
s.head = wake.next
|
|
if s.head == nil {
|
|
s.tail = nil
|
|
}
|
|
}
|
|
unlock(&s.lock)
|
|
if wake != nil {
|
|
wake.next = nil
|
|
goready(wake.g)
|
|
}
|
|
} else {
|
|
// Enqueue itself.
|
|
w := acquireSudog()
|
|
w.g = getg()
|
|
w.nrelease = -1
|
|
w.next = nil
|
|
w.releasetime = 0
|
|
t0 := int64(0)
|
|
if blockprofilerate > 0 {
|
|
t0 = cputicks()
|
|
w.releasetime = -1
|
|
}
|
|
if s.tail == nil {
|
|
s.head = w
|
|
} else {
|
|
s.tail.next = w
|
|
}
|
|
s.tail = w
|
|
goparkunlock(&s.lock, "semacquire", traceEvGoBlockCond)
|
|
if t0 != 0 {
|
|
blockevent(int64(w.releasetime)-t0, 2)
|
|
}
|
|
releaseSudog(w)
|
|
}
|
|
}
|
|
|
|
// syncsemrelease waits for n pairing syncsemacquire on the same semaphore s.
|
|
//go:linkname syncsemrelease sync.runtime_Syncsemrelease
|
|
func syncsemrelease(s *syncSema, n uint32) {
|
|
lock(&s.lock)
|
|
for n > 0 && s.head != nil && s.head.nrelease < 0 {
|
|
// Have pending acquire, satisfy it.
|
|
wake := s.head
|
|
s.head = wake.next
|
|
if s.head == nil {
|
|
s.tail = nil
|
|
}
|
|
if wake.releasetime != 0 {
|
|
wake.releasetime = cputicks()
|
|
}
|
|
wake.next = nil
|
|
goready(wake.g)
|
|
n--
|
|
}
|
|
if n > 0 {
|
|
// enqueue itself
|
|
w := acquireSudog()
|
|
w.g = getg()
|
|
w.nrelease = int32(n)
|
|
w.next = nil
|
|
w.releasetime = 0
|
|
if s.tail == nil {
|
|
s.head = w
|
|
} else {
|
|
s.tail.next = w
|
|
}
|
|
s.tail = w
|
|
goparkunlock(&s.lock, "semarelease", traceEvGoBlockCond)
|
|
releaseSudog(w)
|
|
} else {
|
|
unlock(&s.lock)
|
|
}
|
|
}
|
|
|
|
//go:linkname syncsemcheck sync.runtime_Syncsemcheck
|
|
func syncsemcheck(sz uintptr) {
|
|
if sz != unsafe.Sizeof(syncSema{}) {
|
|
print("runtime: bad syncSema size - sync=", sz, " runtime=", unsafe.Sizeof(syncSema{}), "\n")
|
|
throw("bad syncSema size")
|
|
}
|
|
}
|