| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109 | 
							- // Copyright (C) 2018 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 (
 
- 	"context"
 
- 	"sync"
 
- )
 
- type byteSemaphore struct {
 
- 	max       int
 
- 	available int
 
- 	mut       sync.Mutex
 
- 	cond      *sync.Cond
 
- }
 
- func newByteSemaphore(max int) *byteSemaphore {
 
- 	if max < 0 {
 
- 		max = 0
 
- 	}
 
- 	s := byteSemaphore{
 
- 		max:       max,
 
- 		available: max,
 
- 	}
 
- 	s.cond = sync.NewCond(&s.mut)
 
- 	return &s
 
- }
 
- func (s *byteSemaphore) takeWithContext(ctx context.Context, bytes int) error {
 
- 	done := make(chan struct{})
 
- 	var err error
 
- 	go func() {
 
- 		err = s.takeInner(ctx, bytes)
 
- 		close(done)
 
- 	}()
 
- 	select {
 
- 	case <-done:
 
- 	case <-ctx.Done():
 
- 		s.cond.Broadcast()
 
- 		<-done
 
- 	}
 
- 	return err
 
- }
 
- func (s *byteSemaphore) take(bytes int) {
 
- 	_ = s.takeInner(context.Background(), bytes)
 
- }
 
- func (s *byteSemaphore) takeInner(ctx context.Context, bytes int) error {
 
- 	// Checking context for bytes <= s.available is required for testing and doesn't do any harm.
 
- 	select {
 
- 	case <-ctx.Done():
 
- 		return ctx.Err()
 
- 	default:
 
- 	}
 
- 	s.mut.Lock()
 
- 	defer s.mut.Unlock()
 
- 	if bytes > s.max {
 
- 		bytes = s.max
 
- 	}
 
- 	for bytes > s.available {
 
- 		s.cond.Wait()
 
- 		select {
 
- 		case <-ctx.Done():
 
- 			return ctx.Err()
 
- 		default:
 
- 		}
 
- 		if bytes > s.max {
 
- 			bytes = s.max
 
- 		}
 
- 	}
 
- 	s.available -= bytes
 
- 	return nil
 
- }
 
- func (s *byteSemaphore) give(bytes int) {
 
- 	s.mut.Lock()
 
- 	if bytes > s.max {
 
- 		bytes = s.max
 
- 	}
 
- 	if s.available+bytes > s.max {
 
- 		s.available = s.max
 
- 	} else {
 
- 		s.available += bytes
 
- 	}
 
- 	s.cond.Broadcast()
 
- 	s.mut.Unlock()
 
- }
 
- func (s *byteSemaphore) setCapacity(cap int) {
 
- 	if cap < 0 {
 
- 		cap = 0
 
- 	}
 
- 	s.mut.Lock()
 
- 	diff := cap - s.max
 
- 	s.max = cap
 
- 	s.available += diff
 
- 	if s.available < 0 {
 
- 		s.available = 0
 
- 	} else if s.available > s.max {
 
- 		s.available = s.max
 
- 	}
 
- 	s.cond.Broadcast()
 
- 	s.mut.Unlock()
 
- }
 
 
  |