| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101 | 
							- // 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 https://mozilla.org/MPL/2.0/.
 
- package model
 
- import (
 
- 	"fmt"
 
- 	"path/filepath"
 
- 	"sync"
 
- 	"time"
 
- 	"github.com/pkg/errors"
 
- 	"github.com/syncthing/syncthing/lib/fs"
 
- )
 
- 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()
 
- 				_ = 1 // empty critical section
 
- 				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)
 
- 			}
 
- 		}
 
- 	}()
 
- }
 
- // inWritableDir calls fn(path), while making sure that the directory
 
- // containing `path` is writable for the duration of the call.
 
- func inWritableDir(fn func(string) error, targetFs fs.Filesystem, path string, ignorePerms bool) error {
 
- 	dir := filepath.Dir(path)
 
- 	info, err := targetFs.Stat(dir)
 
- 	if err != nil {
 
- 		return err
 
- 	}
 
- 	if !info.IsDir() {
 
- 		return errors.New("Not a directory: " + path)
 
- 	}
 
- 	if info.Mode()&0200 == 0 {
 
- 		// A non-writeable directory (for this user; we assume that's the
 
- 		// relevant part). Temporarily change the mode so we can delete the
 
- 		// file or directory inside it.
 
- 		if err := targetFs.Chmod(dir, 0755); err == nil {
 
- 			// Chmod succeeded, we should change the permissions back on the way
 
- 			// out. If we fail we log the error as we have irrevocably messed up
 
- 			// at this point. :( (The operation we were called to wrap has
 
- 			// succeeded or failed on its own so returning an error to the
 
- 			// caller is inappropriate.)
 
- 			defer func() {
 
- 				if err := targetFs.Chmod(dir, info.Mode()&fs.ModePerm); err != nil && !fs.IsNotExist(err) {
 
- 					logFn := l.Warnln
 
- 					if ignorePerms {
 
- 						logFn = l.Debugln
 
- 					}
 
- 					logFn("Failed to restore directory permissions after gaining write access:", err)
 
- 				}
 
- 			}()
 
- 		}
 
- 	}
 
- 	return fn(path)
 
- }
 
 
  |