2016-08-14 17:44:20 -07:00
// Copyright 2016 The Gogs Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2016-08-14 17:44:20 -07:00
package sync
import (
"sync"
)
2016-08-30 15:19:53 -07:00
// ExclusivePool is a pool of non-identical instances
2016-08-14 17:44:20 -07:00
// that only one instance with same identity is in the pool at a time.
2016-08-30 15:19:53 -07:00
// In other words, only instances with different identities can be in
// the pool the same time. If another instance with same identity tries
// to get into the pool, it hangs until previous instance left the pool.
2016-08-14 17:44:20 -07:00
//
// This pool is particularly useful for performing tasks on same resource
// on the file system in different goroutines.
2016-08-30 15:19:53 -07:00
type ExclusivePool struct {
2016-08-14 17:44:20 -07:00
lock sync . Mutex
// pool maintains locks for each instance in the pool.
pool map [ string ] * sync . Mutex
// count maintains the number of times an instance with same identity checks in
// to the pool, and should be reduced to 0 (removed from map) by checking out
// with same number of times.
2016-08-17 16:10:07 -07:00
// The purpose of count is to delete lock when count down to 0 and recycle memory
// from map object.
2016-08-14 17:44:20 -07:00
count map [ string ] int
}
2016-08-30 15:19:53 -07:00
// NewExclusivePool initializes and returns a new ExclusivePool object.
func NewExclusivePool ( ) * ExclusivePool {
return & ExclusivePool {
2016-08-14 17:44:20 -07:00
pool : make ( map [ string ] * sync . Mutex ) ,
count : make ( map [ string ] int ) ,
}
}
// CheckIn checks in an instance to the pool and hangs while instance
2017-03-14 20:52:01 -04:00
// with same identity is using the lock.
2016-08-30 15:19:53 -07:00
func ( p * ExclusivePool ) CheckIn ( identity string ) {
2016-08-14 17:44:20 -07:00
p . lock . Lock ( )
lock , has := p . pool [ identity ]
if ! has {
lock = & sync . Mutex { }
p . pool [ identity ] = lock
}
p . count [ identity ] ++
p . lock . Unlock ( )
lock . Lock ( )
}
// CheckOut checks out an instance from the pool and releases the lock
// to let other instances with same identity to grab the lock.
2016-08-30 15:19:53 -07:00
func ( p * ExclusivePool ) CheckOut ( identity string ) {
2016-08-14 17:44:20 -07:00
p . lock . Lock ( )
defer p . lock . Unlock ( )
p . pool [ identity ] . Unlock ( )
if p . count [ identity ] == 1 {
delete ( p . pool , identity )
delete ( p . count , identity )
} else {
p . count [ identity ] --
}
}