Track RUnlockers while locking a RWMutex
This commit is contained in:
parent
433b923ea7
commit
e041a4d212
@ -10,7 +10,9 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -40,7 +42,9 @@ func NewMutex() Mutex {
|
|||||||
|
|
||||||
func NewRWMutex() RWMutex {
|
func NewRWMutex() RWMutex {
|
||||||
if debug {
|
if debug {
|
||||||
return &loggedRWMutex{}
|
return &loggedRWMutex{
|
||||||
|
unlockers: make([]string, 0),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return &sync.RWMutex{}
|
return &sync.RWMutex{}
|
||||||
}
|
}
|
||||||
@ -76,20 +80,28 @@ type loggedRWMutex struct {
|
|||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
start time.Time
|
start time.Time
|
||||||
lockedAt string
|
lockedAt string
|
||||||
|
|
||||||
|
logUnlockers uint32
|
||||||
|
|
||||||
|
unlockers []string
|
||||||
|
unlockersMut sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *loggedRWMutex) Lock() {
|
func (m *loggedRWMutex) Lock() {
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
|
|
||||||
|
atomic.StoreUint32(&m.logUnlockers, 1)
|
||||||
m.RWMutex.Lock()
|
m.RWMutex.Lock()
|
||||||
|
m.logUnlockers = 0
|
||||||
|
|
||||||
m.start = time.Now()
|
m.start = time.Now()
|
||||||
duration := m.start.Sub(start)
|
duration := m.start.Sub(start)
|
||||||
|
|
||||||
m.lockedAt = getCaller()
|
m.lockedAt = getCaller()
|
||||||
if duration > threshold {
|
if duration > threshold {
|
||||||
l.Debugf("RWMutex took %v to lock. Locked at %s", duration, m.lockedAt)
|
l.Debugf("RWMutex took %v to lock. Locked at %s. RUnlockers while locking: %s", duration, m.lockedAt, strings.Join(m.unlockers, ", "))
|
||||||
}
|
}
|
||||||
|
m.unlockers = m.unlockers[0:]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *loggedRWMutex) Unlock() {
|
func (m *loggedRWMutex) Unlock() {
|
||||||
@ -100,6 +112,15 @@ func (m *loggedRWMutex) Unlock() {
|
|||||||
m.RWMutex.Unlock()
|
m.RWMutex.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *loggedRWMutex) RUnlock() {
|
||||||
|
if atomic.LoadUint32(&m.logUnlockers) == 1 {
|
||||||
|
m.unlockersMut.Lock()
|
||||||
|
m.unlockers = append(m.unlockers, getCaller())
|
||||||
|
m.unlockersMut.Unlock()
|
||||||
|
}
|
||||||
|
m.RWMutex.RUnlock()
|
||||||
|
}
|
||||||
|
|
||||||
type loggedWaitGroup struct {
|
type loggedWaitGroup struct {
|
||||||
sync.WaitGroup
|
sync.WaitGroup
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user