package main import ( "sync" ) const ( MaxIndex = ^uint(0) ) type ( // Counter is responsible for operating queue read and write indexes Counter struct { WriteIndex uint // Number of the record last written to the queue ReadIndex uint // Number of the record last read from the queue // If WriteIndex is greater than ReadIndex then there are unread messages // If WriteIndex is less tham ReadIndex then MaxIndex was reached Read chan uint mutex sync.Mutex streaming *sync.Cond } ) func NewCounter(wi, ri uint) *Counter { m := &sync.Mutex{} m.Lock() c := &Counter{ WriteIndex: wi, ReadIndex: ri, Read: make(chan uint), streaming: sync.NewCond(m), } go c.Stream() return c } func (c *Counter) Write(proc func(i uint) bool) { c.mutex.Lock() defer c.mutex.Unlock() ok := proc(c.WriteIndex + 1) if ok { c.WriteIndex++ c.streaming.Signal() } } func (c *Counter) Distance() uint { d := c.WriteIndex - c.ReadIndex if d < 0 { d += MaxIndex } return d } func (c *Counter) Stream() { for { if c.Distance() == 0 { c.streaming.Wait() } c.Read <- c.ReadIndex + 1 c.ReadIndex++ } }