| 
									
										
										
										
											2016-11-09 02:01:56 +01:00
										 |  |  | // Copyright 2016 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Package flowcontrol implements a client side flow control mechanism | 
					
						
							|  |  |  | package flowcontrol | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"sync" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common/mclock" | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const rcConst = 1000000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type cmNode struct { | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 	node                         *ClientNode | 
					
						
							|  |  |  | 	lastUpdate                   mclock.AbsTime | 
					
						
							|  |  |  | 	serving, recharging          bool | 
					
						
							|  |  |  | 	rcWeight                     uint64 | 
					
						
							|  |  |  | 	rcValue, rcDelta, startValue int64 | 
					
						
							|  |  |  | 	finishRecharge               mclock.AbsTime | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | func (node *cmNode) update(time mclock.AbsTime) { | 
					
						
							|  |  |  | 	dt := int64(time - node.lastUpdate) | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	node.rcValue += node.rcDelta * dt / rcConst | 
					
						
							|  |  |  | 	node.lastUpdate = time | 
					
						
							|  |  |  | 	if node.recharging && time >= node.finishRecharge { | 
					
						
							|  |  |  | 		node.recharging = false | 
					
						
							|  |  |  | 		node.rcDelta = 0 | 
					
						
							|  |  |  | 		node.rcValue = 0 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (node *cmNode) set(serving bool, simReqCnt, sumWeight uint64) { | 
					
						
							|  |  |  | 	if node.serving && !serving { | 
					
						
							|  |  |  | 		node.recharging = true | 
					
						
							|  |  |  | 		sumWeight += node.rcWeight | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	node.serving = serving | 
					
						
							|  |  |  | 	if node.recharging && serving { | 
					
						
							|  |  |  | 		node.recharging = false | 
					
						
							|  |  |  | 		sumWeight -= node.rcWeight | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	node.rcDelta = 0 | 
					
						
							|  |  |  | 	if serving { | 
					
						
							|  |  |  | 		node.rcDelta = int64(rcConst / simReqCnt) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if node.recharging { | 
					
						
							|  |  |  | 		node.rcDelta = -int64(node.node.cm.rcRecharge * node.rcWeight / sumWeight) | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 		node.finishRecharge = node.lastUpdate + mclock.AbsTime(node.rcValue*rcConst/(-node.rcDelta)) | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type ClientManager struct { | 
					
						
							|  |  |  | 	lock                             sync.Mutex | 
					
						
							|  |  |  | 	nodes                            map[*cmNode]struct{} | 
					
						
							|  |  |  | 	simReqCnt, sumWeight, rcSumValue uint64 | 
					
						
							|  |  |  | 	maxSimReq, maxRcSum              uint64 | 
					
						
							|  |  |  | 	rcRecharge                       uint64 | 
					
						
							|  |  |  | 	resumeQueue                      chan chan bool | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 	time                             mclock.AbsTime | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NewClientManager(rcTarget, maxSimReq, maxRcSum uint64) *ClientManager { | 
					
						
							|  |  |  | 	cm := &ClientManager{ | 
					
						
							|  |  |  | 		nodes:       make(map[*cmNode]struct{}), | 
					
						
							|  |  |  | 		resumeQueue: make(chan chan bool), | 
					
						
							|  |  |  | 		rcRecharge:  rcConst * rcConst / (100*rcConst/rcTarget - rcConst), | 
					
						
							|  |  |  | 		maxSimReq:   maxSimReq, | 
					
						
							|  |  |  | 		maxRcSum:    maxRcSum, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	go cm.queueProc() | 
					
						
							|  |  |  | 	return cm | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *ClientManager) Stop() { | 
					
						
							|  |  |  | 	self.lock.Lock() | 
					
						
							|  |  |  | 	defer self.lock.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// signal any waiting accept routines to return false | 
					
						
							|  |  |  | 	self.nodes = make(map[*cmNode]struct{}) | 
					
						
							|  |  |  | 	close(self.resumeQueue) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *ClientManager) addNode(cnode *ClientNode) *cmNode { | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 	time := mclock.Now() | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	node := &cmNode{ | 
					
						
							|  |  |  | 		node:           cnode, | 
					
						
							|  |  |  | 		lastUpdate:     time, | 
					
						
							|  |  |  | 		finishRecharge: time, | 
					
						
							|  |  |  | 		rcWeight:       1, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	self.lock.Lock() | 
					
						
							|  |  |  | 	defer self.lock.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	self.nodes[node] = struct{}{} | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 	self.update(mclock.Now()) | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	return node | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *ClientManager) removeNode(node *cmNode) { | 
					
						
							|  |  |  | 	self.lock.Lock() | 
					
						
							|  |  |  | 	defer self.lock.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 	time := mclock.Now() | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	self.stop(node, time) | 
					
						
							|  |  |  | 	delete(self.nodes, node) | 
					
						
							|  |  |  | 	self.update(time) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // recalc sumWeight | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | func (self *ClientManager) updateNodes(time mclock.AbsTime) (rce bool) { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	var sumWeight, rcSum uint64 | 
					
						
							| 
									
										
										
										
											2017-01-06 15:52:03 +01:00
										 |  |  | 	for node := range self.nodes { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 		rc := node.recharging | 
					
						
							|  |  |  | 		node.update(time) | 
					
						
							|  |  |  | 		if rc && !node.recharging { | 
					
						
							|  |  |  | 			rce = true | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if node.recharging { | 
					
						
							|  |  |  | 			sumWeight += node.rcWeight | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		rcSum += uint64(node.rcValue) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	self.sumWeight = sumWeight | 
					
						
							|  |  |  | 	self.rcSumValue = rcSum | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | func (self *ClientManager) update(time mclock.AbsTime) { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	for { | 
					
						
							|  |  |  | 		firstTime := time | 
					
						
							| 
									
										
										
										
											2017-01-06 15:52:03 +01:00
										 |  |  | 		for node := range self.nodes { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 			if node.recharging && node.finishRecharge < firstTime { | 
					
						
							|  |  |  | 				firstTime = node.finishRecharge | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if self.updateNodes(firstTime) { | 
					
						
							| 
									
										
										
										
											2017-01-06 15:52:03 +01:00
										 |  |  | 			for node := range self.nodes { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 				if node.recharging { | 
					
						
							|  |  |  | 					node.set(node.serving, self.simReqCnt, self.sumWeight) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			self.time = time | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *ClientManager) canStartReq() bool { | 
					
						
							|  |  |  | 	return self.simReqCnt < self.maxSimReq && self.rcSumValue < self.maxRcSum | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *ClientManager) queueProc() { | 
					
						
							|  |  |  | 	for rc := range self.resumeQueue { | 
					
						
							|  |  |  | 		for { | 
					
						
							|  |  |  | 			time.Sleep(time.Millisecond * 10) | 
					
						
							|  |  |  | 			self.lock.Lock() | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | 			self.update(mclock.Now()) | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 			cs := self.canStartReq() | 
					
						
							|  |  |  | 			self.lock.Unlock() | 
					
						
							|  |  |  | 			if cs { | 
					
						
							|  |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		close(rc) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | func (self *ClientManager) accept(node *cmNode, time mclock.AbsTime) bool { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	self.lock.Lock() | 
					
						
							|  |  |  | 	defer self.lock.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	self.update(time) | 
					
						
							|  |  |  | 	if !self.canStartReq() { | 
					
						
							|  |  |  | 		resume := make(chan bool) | 
					
						
							|  |  |  | 		self.lock.Unlock() | 
					
						
							|  |  |  | 		self.resumeQueue <- resume | 
					
						
							|  |  |  | 		<-resume | 
					
						
							|  |  |  | 		self.lock.Lock() | 
					
						
							|  |  |  | 		if _, ok := self.nodes[node]; !ok { | 
					
						
							|  |  |  | 			return false // reject if node has been removed or manager has been stopped | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	self.simReqCnt++ | 
					
						
							|  |  |  | 	node.set(true, self.simReqCnt, self.sumWeight) | 
					
						
							|  |  |  | 	node.startValue = node.rcValue | 
					
						
							|  |  |  | 	self.update(self.time) | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | func (self *ClientManager) stop(node *cmNode, time mclock.AbsTime) { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	if node.serving { | 
					
						
							|  |  |  | 		self.update(time) | 
					
						
							|  |  |  | 		self.simReqCnt-- | 
					
						
							|  |  |  | 		node.set(false, self.simReqCnt, self.sumWeight) | 
					
						
							|  |  |  | 		self.update(time) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 11:13:52 +01:00
										 |  |  | func (self *ClientManager) processed(node *cmNode, time mclock.AbsTime) (rcValue, rcCost uint64) { | 
					
						
							| 
									
										
										
										
											2016-10-14 05:51:29 +02:00
										 |  |  | 	self.lock.Lock() | 
					
						
							|  |  |  | 	defer self.lock.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	self.stop(node, time) | 
					
						
							|  |  |  | 	return uint64(node.rcValue), uint64(node.rcValue - node.startValue) | 
					
						
							|  |  |  | } |