mirror of
https://github.com/talent-plan/tinykv.git
synced 2025-01-28 13:20:27 +08:00
5e089a2cd1
Signed-off-by: Connor <zbk602423539@gmail.com> Co-authored-by: Nick Cameron <nrc@ncameron.org> Co-authored-by: linning <linningde25@gmail.com> Co-authored-by: YangKeao <keao.yang@yahoo.com> Co-authored-by: andylokandy <andylokandy@hotmail.com> Co-authored-by: Iosmanthus Teng <myosmanthustree@gmail.com>
131 lines
3.2 KiB
Go
131 lines
3.2 KiB
Go
package raftstore
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/pingcap-incubator/tinykv/kv/config"
|
|
"github.com/pingcap-incubator/tinykv/kv/raftstore/message"
|
|
)
|
|
|
|
type ticker struct {
|
|
regionID uint64
|
|
tick int64
|
|
schedules []tickSchedule
|
|
}
|
|
|
|
type tickSchedule struct {
|
|
runAt int64
|
|
interval int64
|
|
}
|
|
|
|
func newTicker(regionID uint64, cfg *config.Config) *ticker {
|
|
baseInterval := cfg.RaftBaseTickInterval
|
|
t := &ticker{
|
|
regionID: regionID,
|
|
schedules: make([]tickSchedule, 6),
|
|
}
|
|
t.schedules[int(PeerTickRaft)].interval = 1
|
|
t.schedules[int(PeerTickRaftLogGC)].interval = int64(cfg.RaftLogGCTickInterval / baseInterval)
|
|
t.schedules[int(PeerTickSplitRegionCheck)].interval = int64(cfg.SplitRegionCheckTickInterval / baseInterval)
|
|
t.schedules[int(PeerTickSchedulerHeartbeat)].interval = int64(cfg.SchedulerHeartbeatTickInterval / baseInterval)
|
|
return t
|
|
}
|
|
|
|
const SnapMgrGcTickInterval = 1 * time.Minute
|
|
|
|
func newStoreTicker(cfg *config.Config) *ticker {
|
|
baseInterval := cfg.RaftBaseTickInterval
|
|
t := &ticker{
|
|
schedules: make([]tickSchedule, 4),
|
|
}
|
|
t.schedules[int(StoreTickSchedulerStoreHeartbeat)].interval = int64(cfg.SchedulerStoreHeartbeatTickInterval / baseInterval)
|
|
t.schedules[int(StoreTickSnapGC)].interval = int64(SnapMgrGcTickInterval / baseInterval)
|
|
return t
|
|
}
|
|
|
|
// tickClock should be called when peerMsgHandler received tick message.
|
|
func (t *ticker) tickClock() {
|
|
t.tick++
|
|
}
|
|
|
|
// schedule arrange the next run for the PeerTick.
|
|
func (t *ticker) schedule(tp PeerTick) {
|
|
sched := &t.schedules[int(tp)]
|
|
if sched.interval <= 0 {
|
|
sched.runAt = -1
|
|
return
|
|
}
|
|
sched.runAt = t.tick + sched.interval
|
|
}
|
|
|
|
// isOnTick checks if the PeerTick should run.
|
|
func (t *ticker) isOnTick(tp PeerTick) bool {
|
|
sched := &t.schedules[int(tp)]
|
|
return sched.runAt == t.tick
|
|
}
|
|
|
|
func (t *ticker) isOnStoreTick(tp StoreTick) bool {
|
|
sched := &t.schedules[int(tp)]
|
|
return sched.runAt == t.tick
|
|
}
|
|
|
|
func (t *ticker) scheduleStore(tp StoreTick) {
|
|
sched := &t.schedules[int(tp)]
|
|
if sched.interval <= 0 {
|
|
sched.runAt = -1
|
|
return
|
|
}
|
|
sched.runAt = t.tick + sched.interval
|
|
}
|
|
|
|
type tickDriver struct {
|
|
baseTickInterval time.Duration
|
|
newRegionCh chan uint64
|
|
regions map[uint64]struct{}
|
|
router *router
|
|
storeTicker *ticker
|
|
}
|
|
|
|
func newTickDriver(baseTickInterval time.Duration, router *router, storeTicker *ticker) *tickDriver {
|
|
return &tickDriver{
|
|
baseTickInterval: baseTickInterval,
|
|
newRegionCh: make(chan uint64),
|
|
regions: make(map[uint64]struct{}),
|
|
router: router,
|
|
storeTicker: storeTicker,
|
|
}
|
|
}
|
|
|
|
func (r *tickDriver) run() {
|
|
timer := time.Tick(r.baseTickInterval)
|
|
for {
|
|
select {
|
|
case <-timer:
|
|
for regionID := range r.regions {
|
|
if r.router.send(regionID, message.NewPeerMsg(message.MsgTypeTick, regionID, nil)) != nil {
|
|
delete(r.regions, regionID)
|
|
}
|
|
}
|
|
r.tickStore()
|
|
case regionID, ok := <-r.newRegionCh:
|
|
if !ok {
|
|
return
|
|
}
|
|
r.regions[regionID] = struct{}{}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (r *tickDriver) stop() {
|
|
close(r.newRegionCh)
|
|
}
|
|
|
|
func (r *tickDriver) tickStore() {
|
|
r.storeTicker.tickClock()
|
|
for i := range r.storeTicker.schedules {
|
|
if r.storeTicker.isOnStoreTick(StoreTick(i)) {
|
|
r.router.sendStore(message.NewMsg(message.MsgTypeStoreTick, StoreTick(i)))
|
|
}
|
|
}
|
|
}
|