Priority queue for tasks
This commit is contained in:
parent
1bbfb7cffd
commit
ced1d6e087
36
request_queue.go
Normal file
36
request_queue.go
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
package sectorstorage
|
||||||
|
|
||||||
|
import "container/heap"
|
||||||
|
|
||||||
|
type requestQueue []*workerRequest
|
||||||
|
|
||||||
|
func (q requestQueue) Len() int { return len(q) }
|
||||||
|
|
||||||
|
func (q requestQueue) Less(i, j int) bool {
|
||||||
|
return q[i].taskType.Less(q[j].taskType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q requestQueue) Swap(i, j int) {
|
||||||
|
q[i], q[j] = q[j], q[i]
|
||||||
|
q[i].index = i
|
||||||
|
q[j].index = j
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *requestQueue) Push(x interface{}) {
|
||||||
|
n := len(*q)
|
||||||
|
item := x.(*workerRequest)
|
||||||
|
item.index = n
|
||||||
|
*q = append(*q, item)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *requestQueue) Pop() interface{} {
|
||||||
|
old := *q
|
||||||
|
n := len(old)
|
||||||
|
item := old[n-1]
|
||||||
|
old[n-1] = nil // avoid memory leak
|
||||||
|
item.index = -1 // for safety
|
||||||
|
*q = old[0 : n-1]
|
||||||
|
return item
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ heap.Interface = &requestQueue{}
|
30
request_queue_test.go
Normal file
30
request_queue_test.go
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
package sectorstorage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"container/heap"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/filecoin-project/sector-storage/sealtasks"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRequestQueue(t *testing.T) {
|
||||||
|
rq := &requestQueue{}
|
||||||
|
|
||||||
|
heap.Push(rq, &workerRequest{taskType: sealtasks.TTAddPiece})
|
||||||
|
heap.Push(rq, &workerRequest{taskType: sealtasks.TTPreCommit1})
|
||||||
|
heap.Push(rq, &workerRequest{taskType: sealtasks.TTPreCommit2})
|
||||||
|
heap.Push(rq, &workerRequest{taskType: sealtasks.TTPreCommit1})
|
||||||
|
heap.Push(rq, &workerRequest{taskType: sealtasks.TTAddPiece})
|
||||||
|
|
||||||
|
pt := heap.Pop(rq).(*workerRequest)
|
||||||
|
|
||||||
|
if pt.taskType != sealtasks.TTPreCommit2 {
|
||||||
|
t.Error("expected precommit2, got", pt.taskType)
|
||||||
|
}
|
||||||
|
|
||||||
|
pt = heap.Pop(rq).(*workerRequest)
|
||||||
|
|
||||||
|
if pt.taskType != sealtasks.TTPreCommit1 {
|
||||||
|
t.Error("expected precommit1, got", pt.taskType)
|
||||||
|
}
|
||||||
|
}
|
25
sched.go
25
sched.go
@ -1,7 +1,7 @@
|
|||||||
package sectorstorage
|
package sectorstorage
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"container/list"
|
"container/heap"
|
||||||
"context"
|
"context"
|
||||||
"sort"
|
"sort"
|
||||||
"sync"
|
"sync"
|
||||||
@ -41,7 +41,7 @@ type scheduler struct {
|
|||||||
workerFree chan WorkerID
|
workerFree chan WorkerID
|
||||||
closing chan struct{}
|
closing chan struct{}
|
||||||
|
|
||||||
schedQueue *list.List // List[*workerRequest]
|
schedQueue *requestQueue
|
||||||
}
|
}
|
||||||
|
|
||||||
func newScheduler(spt abi.RegisteredProof) *scheduler {
|
func newScheduler(spt abi.RegisteredProof) *scheduler {
|
||||||
@ -60,7 +60,7 @@ func newScheduler(spt abi.RegisteredProof) *scheduler {
|
|||||||
workerFree: make(chan WorkerID),
|
workerFree: make(chan WorkerID),
|
||||||
closing: make(chan struct{}),
|
closing: make(chan struct{}),
|
||||||
|
|
||||||
schedQueue: list.New(),
|
schedQueue: &requestQueue{},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -101,6 +101,8 @@ type workerRequest struct {
|
|||||||
prepare WorkerAction
|
prepare WorkerAction
|
||||||
work WorkerAction
|
work WorkerAction
|
||||||
|
|
||||||
|
index int // The index of the item in the heap.
|
||||||
|
|
||||||
ret chan<- workerResponse
|
ret chan<- workerResponse
|
||||||
ctx context.Context
|
ctx context.Context
|
||||||
}
|
}
|
||||||
@ -154,7 +156,7 @@ func (sh *scheduler) runSched() {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
sh.schedQueue.PushBack(req)
|
heap.Push(sh.schedQueue, req)
|
||||||
case wid := <-sh.workerFree:
|
case wid := <-sh.workerFree:
|
||||||
sh.onWorkerFreed(wid)
|
sh.onWorkerFreed(wid)
|
||||||
case <-sh.closing:
|
case <-sh.closing:
|
||||||
@ -173,8 +175,8 @@ func (sh *scheduler) onWorkerFreed(wid WorkerID) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
for e := sh.schedQueue.Front(); e != nil; e = e.Next() {
|
for i := 0; i < sh.schedQueue.Len(); i++ {
|
||||||
req := e.Value.(*workerRequest)
|
req := (*sh.schedQueue)[i]
|
||||||
|
|
||||||
ok, err := req.sel.Ok(req.ctx, req.taskType, w)
|
ok, err := req.sel.Ok(req.ctx, req.taskType, w)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -193,15 +195,8 @@ func (sh *scheduler) onWorkerFreed(wid WorkerID) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if scheduled {
|
if scheduled {
|
||||||
pe := e.Prev()
|
heap.Remove(sh.schedQueue, i)
|
||||||
sh.schedQueue.Remove(e)
|
i--
|
||||||
if pe == nil {
|
|
||||||
pe = sh.schedQueue.Front()
|
|
||||||
}
|
|
||||||
if pe == nil {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
e = pe
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,3 +13,17 @@ const (
|
|||||||
|
|
||||||
TTFetch TaskType = "seal/v0/fetch"
|
TTFetch TaskType = "seal/v0/fetch"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var order = map[TaskType]int{
|
||||||
|
TTAddPiece: 7,
|
||||||
|
TTPreCommit1: 6,
|
||||||
|
TTPreCommit2: 5,
|
||||||
|
TTCommit2: 4,
|
||||||
|
TTCommit1: 3,
|
||||||
|
TTFetch: 2,
|
||||||
|
TTFinalize: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a TaskType) Less(b TaskType) bool {
|
||||||
|
return order[a] < order[b]
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user