|
|
|
package stario
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type WaitGroup struct {
|
|
|
|
wg *sync.WaitGroup
|
|
|
|
maxCount uint32
|
|
|
|
allCount uint32
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewWaitGroup(maxCount int) WaitGroup {
|
|
|
|
return WaitGroup{wg: &sync.WaitGroup{}, maxCount: uint32(maxCount)}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *WaitGroup) Add(delta int) {
|
|
|
|
var Udelta uint32
|
|
|
|
if delta < 0 {
|
|
|
|
Udelta = uint32(-delta - 1)
|
|
|
|
} else {
|
|
|
|
Udelta = uint32(delta)
|
|
|
|
}
|
|
|
|
for {
|
|
|
|
allC := atomic.LoadUint32(&w.allCount)
|
|
|
|
if atomic.LoadUint32(&w.maxCount) == 0 || atomic.LoadUint32(&w.maxCount) >= allC+uint32(delta) {
|
|
|
|
if delta < 0 {
|
|
|
|
atomic.AddUint32(&w.allCount, ^uint32(Udelta))
|
|
|
|
} else {
|
|
|
|
atomic.AddUint32(&w.allCount, uint32(Udelta))
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
time.Sleep(time.Microsecond)
|
|
|
|
}
|
|
|
|
w.wg.Add(delta)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *WaitGroup) Done() {
|
|
|
|
w.Add(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *WaitGroup) Wait() {
|
|
|
|
w.wg.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *WaitGroup) GetMaxWaitNum() int {
|
|
|
|
return int(atomic.LoadUint32(&w.maxCount))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *WaitGroup) SetMaxWaitNum(num int) {
|
|
|
|
atomic.AddUint32(&w.maxCount, uint32(num))
|
|
|
|
}
|