| 
									
										
										
										
											2016-08-14 17:44:20 -07:00
										 |  |  | // Copyright 2016 The Gogs Authors. All rights reserved. | 
					
						
							|  |  |  | // Use of this source code is governed by a MIT-style | 
					
						
							|  |  |  | // license that can be found in the LICENSE file. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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]-- | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |