mirror of
https://github.com/octoleo/syncthing.git
synced 2024-11-09 14:50:56 +00:00
162 lines
3.0 KiB
Go
162 lines
3.0 KiB
Go
// Copyright (C) 2015 The Syncthing Authors.
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
package sync
|
|
|
|
import (
|
|
"fmt"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
)
|
|
|
|
type Mutex interface {
|
|
Lock()
|
|
Unlock()
|
|
}
|
|
|
|
type RWMutex interface {
|
|
Mutex
|
|
RLock()
|
|
RUnlock()
|
|
}
|
|
|
|
type WaitGroup interface {
|
|
Add(int)
|
|
Done()
|
|
Wait()
|
|
}
|
|
|
|
func NewMutex() Mutex {
|
|
if debug {
|
|
return &loggedMutex{}
|
|
}
|
|
return &sync.Mutex{}
|
|
}
|
|
|
|
func NewRWMutex() RWMutex {
|
|
if debug {
|
|
return &loggedRWMutex{
|
|
unlockers: make([]string, 0),
|
|
}
|
|
}
|
|
return &sync.RWMutex{}
|
|
}
|
|
|
|
func NewWaitGroup() WaitGroup {
|
|
if debug {
|
|
return &loggedWaitGroup{}
|
|
}
|
|
return &sync.WaitGroup{}
|
|
}
|
|
|
|
type loggedMutex struct {
|
|
sync.Mutex
|
|
start time.Time
|
|
lockedAt string
|
|
goid int
|
|
}
|
|
|
|
func (m *loggedMutex) Lock() {
|
|
m.Mutex.Lock()
|
|
m.start = time.Now()
|
|
m.lockedAt = getCaller()
|
|
m.goid = goid()
|
|
}
|
|
|
|
func (m *loggedMutex) Unlock() {
|
|
duration := time.Now().Sub(m.start)
|
|
if duration >= threshold {
|
|
l.Debugf("Mutex held for %v. Locked at %s unlocked at %s", duration, m.lockedAt, getCaller())
|
|
}
|
|
m.Mutex.Unlock()
|
|
}
|
|
|
|
func (m *loggedMutex) Holder() (string, int) {
|
|
return m.lockedAt, m.goid
|
|
}
|
|
|
|
type loggedRWMutex struct {
|
|
sync.RWMutex
|
|
start time.Time
|
|
lockedAt string
|
|
goid int
|
|
|
|
logUnlockers uint32
|
|
|
|
unlockers []string
|
|
unlockersMut sync.Mutex
|
|
}
|
|
|
|
func (m *loggedRWMutex) Lock() {
|
|
start := time.Now()
|
|
|
|
atomic.StoreUint32(&m.logUnlockers, 1)
|
|
m.RWMutex.Lock()
|
|
m.logUnlockers = 0
|
|
|
|
m.start = time.Now()
|
|
duration := m.start.Sub(start)
|
|
|
|
m.lockedAt = getCaller()
|
|
m.goid = goid()
|
|
if duration > threshold {
|
|
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() {
|
|
duration := time.Now().Sub(m.start)
|
|
if duration >= threshold {
|
|
l.Debugf("RWMutex held for %v. Locked at %s: unlocked at %s", duration, m.lockedAt, getCaller())
|
|
}
|
|
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 {
|
|
sync.WaitGroup
|
|
}
|
|
|
|
func (wg *loggedWaitGroup) Wait() {
|
|
start := time.Now()
|
|
wg.WaitGroup.Wait()
|
|
duration := time.Now().Sub(start)
|
|
if duration >= threshold {
|
|
l.Debugf("WaitGroup took %v at %s", duration, getCaller())
|
|
}
|
|
}
|
|
|
|
func getCaller() string {
|
|
_, file, line, _ := runtime.Caller(2)
|
|
file = filepath.Join(filepath.Base(filepath.Dir(file)), filepath.Base(file))
|
|
return fmt.Sprintf("%s:%d", file, line)
|
|
}
|
|
|
|
func goid() int {
|
|
var buf [64]byte
|
|
n := runtime.Stack(buf[:], false)
|
|
idField := strings.Fields(strings.TrimPrefix(string(buf[:n]), "goroutine "))[0]
|
|
id, err := strconv.Atoi(idField)
|
|
if err != nil {
|
|
return -1
|
|
}
|
|
return id
|
|
}
|