mirror of
				https://github.com/containers/skopeo.git
				synced 2025-10-22 11:44:05 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			174 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package mpb
 | |
| 
 | |
| import "container/heap"
 | |
| 
 | |
| type heapManager chan heapRequest
 | |
| 
 | |
| type heapCmd int
 | |
| 
 | |
| const (
 | |
| 	h_sync heapCmd = iota
 | |
| 	h_push
 | |
| 	h_iter
 | |
| 	h_drain
 | |
| 	h_fix
 | |
| 	h_state
 | |
| 	h_end
 | |
| )
 | |
| 
 | |
| type heapRequest struct {
 | |
| 	cmd  heapCmd
 | |
| 	data interface{}
 | |
| }
 | |
| 
 | |
| type iterData struct {
 | |
| 	iter chan<- *Bar
 | |
| 	drop <-chan struct{}
 | |
| }
 | |
| 
 | |
| type pushData struct {
 | |
| 	bar  *Bar
 | |
| 	sync bool
 | |
| }
 | |
| 
 | |
| type fixData struct {
 | |
| 	bar      *Bar
 | |
| 	priority int
 | |
| 	lazy     bool
 | |
| }
 | |
| 
 | |
| func (m heapManager) run() {
 | |
| 	var bHeap priorityQueue
 | |
| 	var pMatrix, aMatrix map[int][]chan int
 | |
| 
 | |
| 	var l int
 | |
| 	var sync bool
 | |
| 
 | |
| 	for req := range m {
 | |
| 		switch req.cmd {
 | |
| 		case h_push:
 | |
| 			data := req.data.(pushData)
 | |
| 			heap.Push(&bHeap, data.bar)
 | |
| 			if !sync {
 | |
| 				sync = data.sync
 | |
| 			}
 | |
| 		case h_sync:
 | |
| 			if sync || l != bHeap.Len() {
 | |
| 				pMatrix = make(map[int][]chan int)
 | |
| 				aMatrix = make(map[int][]chan int)
 | |
| 				for _, b := range bHeap {
 | |
| 					table := b.wSyncTable()
 | |
| 					for i, ch := range table[0] {
 | |
| 						pMatrix[i] = append(pMatrix[i], ch)
 | |
| 					}
 | |
| 					for i, ch := range table[1] {
 | |
| 						aMatrix[i] = append(aMatrix[i], ch)
 | |
| 					}
 | |
| 				}
 | |
| 				sync = false
 | |
| 				l = bHeap.Len()
 | |
| 			}
 | |
| 			drop := req.data.(<-chan struct{})
 | |
| 			syncWidth(pMatrix, drop)
 | |
| 			syncWidth(aMatrix, drop)
 | |
| 		case h_iter:
 | |
| 			data := req.data.(iterData)
 | |
| 		drop_iter:
 | |
| 			for _, b := range bHeap {
 | |
| 				select {
 | |
| 				case data.iter <- b:
 | |
| 				case <-data.drop:
 | |
| 					break drop_iter
 | |
| 				}
 | |
| 			}
 | |
| 			close(data.iter)
 | |
| 		case h_drain:
 | |
| 			data := req.data.(iterData)
 | |
| 		drop_drain:
 | |
| 			for bHeap.Len() != 0 {
 | |
| 				select {
 | |
| 				case data.iter <- heap.Pop(&bHeap).(*Bar):
 | |
| 				case <-data.drop:
 | |
| 					break drop_drain
 | |
| 				}
 | |
| 			}
 | |
| 			close(data.iter)
 | |
| 		case h_fix:
 | |
| 			data := req.data.(fixData)
 | |
| 			if data.bar.index < 0 {
 | |
| 				break
 | |
| 			}
 | |
| 			data.bar.priority = data.priority
 | |
| 			if !data.lazy {
 | |
| 				heap.Fix(&bHeap, data.bar.index)
 | |
| 			}
 | |
| 		case h_state:
 | |
| 			ch := req.data.(chan<- bool)
 | |
| 			ch <- sync || l != bHeap.Len()
 | |
| 		case h_end:
 | |
| 			ch := req.data.(chan<- interface{})
 | |
| 			if ch != nil {
 | |
| 				go func() {
 | |
| 					ch <- []*Bar(bHeap)
 | |
| 				}()
 | |
| 			}
 | |
| 			close(m)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (m heapManager) sync(drop <-chan struct{}) {
 | |
| 	m <- heapRequest{cmd: h_sync, data: drop}
 | |
| }
 | |
| 
 | |
| func (m heapManager) push(b *Bar, sync bool) {
 | |
| 	data := pushData{b, sync}
 | |
| 	m <- heapRequest{cmd: h_push, data: data}
 | |
| }
 | |
| 
 | |
| func (m heapManager) iter(iter chan<- *Bar, drop <-chan struct{}) {
 | |
| 	data := iterData{iter, drop}
 | |
| 	m <- heapRequest{cmd: h_iter, data: data}
 | |
| }
 | |
| 
 | |
| func (m heapManager) drain(iter chan<- *Bar, drop <-chan struct{}) {
 | |
| 	data := iterData{iter, drop}
 | |
| 	m <- heapRequest{cmd: h_drain, data: data}
 | |
| }
 | |
| 
 | |
| func (m heapManager) fix(b *Bar, priority int, lazy bool) {
 | |
| 	data := fixData{b, priority, lazy}
 | |
| 	m <- heapRequest{cmd: h_fix, data: data}
 | |
| }
 | |
| 
 | |
| func (m heapManager) state(ch chan<- bool) {
 | |
| 	m <- heapRequest{cmd: h_state, data: ch}
 | |
| }
 | |
| 
 | |
| func (m heapManager) end(ch chan<- interface{}) {
 | |
| 	m <- heapRequest{cmd: h_end, data: ch}
 | |
| }
 | |
| 
 | |
| func syncWidth(matrix map[int][]chan int, drop <-chan struct{}) {
 | |
| 	for _, column := range matrix {
 | |
| 		go maxWidthDistributor(column, drop)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func maxWidthDistributor(column []chan int, drop <-chan struct{}) {
 | |
| 	var maxWidth int
 | |
| 	for _, ch := range column {
 | |
| 		select {
 | |
| 		case w := <-ch:
 | |
| 			if w > maxWidth {
 | |
| 				maxWidth = w
 | |
| 			}
 | |
| 		case <-drop:
 | |
| 			return
 | |
| 		}
 | |
| 	}
 | |
| 	for _, ch := range column {
 | |
| 		ch <- maxWidth
 | |
| 	}
 | |
| }
 |