vendor: Mega update all dependencies
GitHub-Pull-Request: https://github.com/syncthing/syncthing/pull/4080
This commit is contained in:
59
vendor/github.com/sasha-s/go-deadlock/deadlock.go
generated
vendored
59
vendor/github.com/sasha-s/go-deadlock/deadlock.go
generated
vendored
@@ -5,10 +5,10 @@ import (
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
"runtime"
|
||||
"strconv"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/petermattis/goid"
|
||||
)
|
||||
|
||||
// Opts control how deadlock detection behaves.
|
||||
@@ -62,12 +62,10 @@ func (m *Mutex) Lock() {
|
||||
// It is allowed for one goroutine to lock a Mutex and then
|
||||
// arrange for another goroutine to unlock it.
|
||||
func (m *Mutex) Unlock() {
|
||||
if Opts.Disable {
|
||||
m.mu.Unlock()
|
||||
return
|
||||
}
|
||||
m.mu.Unlock()
|
||||
PostUnlock(m)
|
||||
if !Opts.Disable {
|
||||
PostUnlock(m)
|
||||
}
|
||||
}
|
||||
|
||||
// An RWMutex is a drop-in replacement for sync.RWMutex.
|
||||
@@ -96,12 +94,10 @@ func (m *RWMutex) Lock() {
|
||||
// goroutine. One goroutine may RLock (Lock) an RWMutex and then
|
||||
// arrange for another goroutine to RUnlock (Unlock) it.
|
||||
func (m *RWMutex) Unlock() {
|
||||
if Opts.Disable {
|
||||
m.mu.Unlock()
|
||||
return
|
||||
}
|
||||
m.mu.Unlock()
|
||||
PostUnlock(m)
|
||||
if !Opts.Disable {
|
||||
PostUnlock(m)
|
||||
}
|
||||
}
|
||||
|
||||
// RLock locks the mutex for reading.
|
||||
@@ -117,12 +113,10 @@ func (m *RWMutex) RLock() {
|
||||
// It is a run-time error if rw is not locked for reading
|
||||
// on entry to RUnlock.
|
||||
func (m *RWMutex) RUnlock() {
|
||||
if Opts.Disable {
|
||||
m.mu.RUnlock()
|
||||
return
|
||||
if !Opts.Disable {
|
||||
PostUnlock(m)
|
||||
}
|
||||
m.mu.RUnlock()
|
||||
PostUnlock(m)
|
||||
}
|
||||
|
||||
// RLocker returns a Locker interface that implements
|
||||
@@ -173,13 +167,13 @@ func lock(lockFn func(), ptr interface{}) {
|
||||
fmt.Fprintf(Opts.LogBuf, "goroutine %v lock %p\n", prev.gid, ptr)
|
||||
printStack(Opts.LogBuf, prev.stack)
|
||||
fmt.Fprintln(Opts.LogBuf, "Have been trying to lock it again for more than", Opts.DeadlockTimeout)
|
||||
fmt.Fprintf(Opts.LogBuf, "goroutine %v lock %p\n", getGID(), ptr)
|
||||
fmt.Fprintf(Opts.LogBuf, "goroutine %v lock %p\n", goid.Get(), ptr)
|
||||
printStack(Opts.LogBuf, callers(2))
|
||||
fmt.Fprintln(Opts.LogBuf)
|
||||
stacks := stacks()
|
||||
grs := bytes.Split(stacks, []byte("\n\n"))
|
||||
for _, g := range grs {
|
||||
if extractGID(g) == prev.gid {
|
||||
if goid.ExtractGID(g) == prev.gid {
|
||||
fmt.Fprintln(Opts.LogBuf, "Here is what goroutine", prev.gid, "doing now")
|
||||
Opts.LogBuf.Write(g)
|
||||
fmt.Fprintln(Opts.LogBuf)
|
||||
@@ -210,7 +204,7 @@ type lockOrder struct {
|
||||
|
||||
type stackGID struct {
|
||||
stack []uintptr
|
||||
gid uint64
|
||||
gid int64
|
||||
}
|
||||
|
||||
type beforeAfter struct {
|
||||
@@ -233,19 +227,20 @@ func newLockOrder() *lockOrder {
|
||||
}
|
||||
|
||||
func (l *lockOrder) PostLock(skip int, p interface{}) {
|
||||
stack := callers(skip)
|
||||
gid := goid.Get()
|
||||
l.mu.Lock()
|
||||
defer l.mu.Unlock()
|
||||
l.cur[p] = stackGID{callers(skip), getGID()}
|
||||
l.cur[p] = stackGID{stack, gid}
|
||||
l.mu.Unlock()
|
||||
}
|
||||
|
||||
func (l *lockOrder) PreLock(skip int, p interface{}) {
|
||||
if Opts.DisableLockOrderDetection {
|
||||
return
|
||||
}
|
||||
l.mu.Lock()
|
||||
defer l.mu.Unlock()
|
||||
stack := callers(skip)
|
||||
gid := getGID()
|
||||
gid := goid.Get()
|
||||
l.mu.Lock()
|
||||
for b, bs := range l.cur {
|
||||
if b == p {
|
||||
continue
|
||||
@@ -272,12 +267,13 @@ func (l *lockOrder) PreLock(skip int, p interface{}) {
|
||||
}
|
||||
}
|
||||
l.cur[p] = stackGID{stack, gid}
|
||||
l.mu.Unlock()
|
||||
}
|
||||
|
||||
func (l *lockOrder) PostUnlock(p interface{}) {
|
||||
l.mu.Lock()
|
||||
defer l.mu.Unlock()
|
||||
delete(l.cur, p)
|
||||
l.mu.Unlock()
|
||||
}
|
||||
|
||||
type rlocker RWMutex
|
||||
@@ -298,17 +294,4 @@ func (l *lockOrder) other(ptr interface{}) {
|
||||
fmt.Fprintln(Opts.LogBuf)
|
||||
}
|
||||
|
||||
// Hacky way of getting a goroutine ID.
|
||||
func getGID() uint64 {
|
||||
b := make([]byte, 64)
|
||||
return extractGID(b[:runtime.Stack(b, false)])
|
||||
}
|
||||
|
||||
func extractGID(stack []byte) uint64 {
|
||||
b := bytes.TrimPrefix(stack, []byte("goroutine "))
|
||||
b = b[:bytes.IndexByte(b, ' ')]
|
||||
gid, _ := strconv.ParseUint(string(b), 10, 64)
|
||||
return gid
|
||||
}
|
||||
|
||||
const header = "POTENTIAL DEADLOCK:"
|
||||
|
||||
127
vendor/github.com/sasha-s/go-deadlock/deadlock_test.go
generated
vendored
127
vendor/github.com/sasha-s/go-deadlock/deadlock_test.go
generated
vendored
@@ -1,127 +0,0 @@
|
||||
package deadlock
|
||||
|
||||
import (
|
||||
"log"
|
||||
"math/rand"
|
||||
"sync"
|
||||
"sync/atomic"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestNoDeadlocks(t *testing.T) {
|
||||
defer restore()()
|
||||
Opts.DeadlockTimeout = time.Millisecond * 5000
|
||||
var a RWMutex
|
||||
var b Mutex
|
||||
var c RWMutex
|
||||
var wg sync.WaitGroup
|
||||
for i := 0; i < 10; i++ {
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
for k := 0; k < 5; k++ {
|
||||
func() {
|
||||
a.Lock()
|
||||
defer a.Unlock()
|
||||
func() {
|
||||
b.Lock()
|
||||
defer b.Unlock()
|
||||
func() {
|
||||
c.RLock()
|
||||
defer c.RUnlock()
|
||||
time.Sleep(time.Duration((1000 + rand.Intn(1000))) * time.Millisecond / 200)
|
||||
}()
|
||||
}()
|
||||
}()
|
||||
}
|
||||
}()
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
for k := 0; k < 5; k++ {
|
||||
func() {
|
||||
a.RLock()
|
||||
defer a.RUnlock()
|
||||
func() {
|
||||
b.Lock()
|
||||
defer b.Unlock()
|
||||
func() {
|
||||
c.Lock()
|
||||
defer c.Unlock()
|
||||
time.Sleep(time.Duration((1000 + rand.Intn(1000))) * time.Millisecond / 200)
|
||||
}()
|
||||
}()
|
||||
}()
|
||||
}
|
||||
}()
|
||||
}
|
||||
wg.Wait()
|
||||
}
|
||||
|
||||
func TestLockOrder(t *testing.T) {
|
||||
defer restore()()
|
||||
Opts.DeadlockTimeout = 0
|
||||
var deadlocks uint32
|
||||
Opts.OnPotentialDeadlock = func() {
|
||||
atomic.AddUint32(&deadlocks, 1)
|
||||
}
|
||||
var a RWMutex
|
||||
var b Mutex
|
||||
var wg sync.WaitGroup
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
a.Lock()
|
||||
b.Lock()
|
||||
b.Unlock()
|
||||
a.Unlock()
|
||||
}()
|
||||
wg.Wait()
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
b.Lock()
|
||||
a.RLock()
|
||||
a.RUnlock()
|
||||
b.Unlock()
|
||||
}()
|
||||
wg.Wait()
|
||||
if deadlocks != 1 {
|
||||
t.Fatalf("expected 1 deadlock, detected %d", deadlocks)
|
||||
}
|
||||
}
|
||||
|
||||
func TestHardDeadlock(t *testing.T) {
|
||||
defer restore()()
|
||||
Opts.DisableLockOrderDetection = true
|
||||
Opts.DeadlockTimeout = time.Millisecond * 20
|
||||
var deadlocks uint32
|
||||
Opts.OnPotentialDeadlock = func() {
|
||||
atomic.AddUint32(&deadlocks, 1)
|
||||
}
|
||||
var mu Mutex
|
||||
mu.Lock()
|
||||
ch := make(chan struct{})
|
||||
go func() {
|
||||
defer close(ch)
|
||||
mu.Lock()
|
||||
defer mu.Unlock()
|
||||
}()
|
||||
select {
|
||||
case <-ch:
|
||||
case <-time.After(time.Millisecond * 100):
|
||||
}
|
||||
if deadlocks != 1 {
|
||||
t.Fatalf("expected 1 deadlock, detected %d", deadlocks)
|
||||
}
|
||||
log.Println("****************")
|
||||
mu.Unlock()
|
||||
}
|
||||
|
||||
func restore() func() {
|
||||
opts := Opts
|
||||
return func() {
|
||||
Opts = opts
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user