avoid goroutine leak

pull/168/head
v2ray 2016-05-16 17:30:00 -07:00
parent 18d75cb7b4
commit 09ea65687c
1 changed files with 24 additions and 20 deletions

View File

@ -16,7 +16,9 @@ import (
) )
const ( const (
DefaultTTL = uint32(3600) DefaultTTL = uint32(3600)
CleanupInterval = time.Second * 120
CleanupThreshold = 512
) )
type ARecord struct { type ARecord struct {
@ -35,9 +37,10 @@ type PendingRequest struct {
type UDPNameServer struct { type UDPNameServer struct {
sync.Mutex sync.Mutex
address v2net.Destination address v2net.Destination
requests map[uint16]*PendingRequest requests map[uint16]*PendingRequest
udpServer *hub.UDPServer udpServer *hub.UDPServer
nextCleanup time.Time
} }
func NewUDPNameServer(address v2net.Destination, dispatcher dispatcher.PacketDispatcher) *UDPNameServer { func NewUDPNameServer(address v2net.Destination, dispatcher dispatcher.PacketDispatcher) *UDPNameServer {
@ -46,35 +49,36 @@ func NewUDPNameServer(address v2net.Destination, dispatcher dispatcher.PacketDis
requests: make(map[uint16]*PendingRequest), requests: make(map[uint16]*PendingRequest),
udpServer: hub.NewUDPServer(dispatcher), udpServer: hub.NewUDPServer(dispatcher),
} }
go s.Cleanup()
return s return s
} }
// @Private // @Private
func (this *UDPNameServer) Cleanup() { func (this *UDPNameServer) Cleanup() {
for { expiredRequests := make([]uint16, 0, 16)
time.Sleep(time.Second * 60) now := time.Now()
expiredRequests := make([]uint16, 0, 16) this.Lock()
now := time.Now() for id, r := range this.requests {
this.Lock() if r.expire.Before(now) {
for id, r := range this.requests { expiredRequests = append(expiredRequests, id)
if r.expire.Before(now) { close(r.response)
expiredRequests = append(expiredRequests, id)
close(r.response)
}
} }
for _, id := range expiredRequests {
delete(this.requests, id)
}
this.Unlock()
expiredRequests = nil
} }
for _, id := range expiredRequests {
delete(this.requests, id)
}
this.Unlock()
expiredRequests = nil
} }
// @Private // @Private
func (this *UDPNameServer) AssignUnusedID(response chan<- *ARecord) uint16 { func (this *UDPNameServer) AssignUnusedID(response chan<- *ARecord) uint16 {
var id uint16 var id uint16
this.Lock() this.Lock()
if len(this.requests) > CleanupThreshold && this.nextCleanup.Before(time.Now()) {
this.nextCleanup = time.Now().Add(CleanupInterval)
go this.Cleanup()
}
for { for {
id = uint16(rand.Intn(65536)) id = uint16(rand.Intn(65536))
if _, found := this.requests[id]; found { if _, found := this.requests[id]; found {