| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960 | 
							- // Copyright (C) 2014 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 model
 
- import (
 
- 	"fmt"
 
- 	"sync"
 
- 	"time"
 
- )
 
- type Holdable interface {
 
- 	Holders() string
 
- }
 
- func newDeadlockDetector(timeout time.Duration) *deadlockDetector {
 
- 	return &deadlockDetector{
 
- 		timeout: timeout,
 
- 		lockers: make(map[string]sync.Locker),
 
- 	}
 
- }
 
- type deadlockDetector struct {
 
- 	timeout time.Duration
 
- 	lockers map[string]sync.Locker
 
- }
 
- func (d *deadlockDetector) Watch(name string, mut sync.Locker) {
 
- 	d.lockers[name] = mut
 
- 	go func() {
 
- 		for {
 
- 			time.Sleep(d.timeout / 4)
 
- 			ok := make(chan bool, 2)
 
- 			go func() {
 
- 				mut.Lock()
 
- 				mut.Unlock()
 
- 				ok <- true
 
- 			}()
 
- 			go func() {
 
- 				time.Sleep(d.timeout)
 
- 				ok <- false
 
- 			}()
 
- 			if r := <-ok; !r {
 
- 				msg := fmt.Sprintf("deadlock detected at %s", name)
 
- 				for otherName, otherMut := range d.lockers {
 
- 					if otherHolder, ok := otherMut.(Holdable); ok {
 
- 						msg += "\n===" + otherName + "===\n" + otherHolder.Holders()
 
- 					}
 
- 				}
 
- 				panic(msg)
 
- 			}
 
- 		}
 
- 	}()
 
- }
 
 
  |