Commit ce5263ff authored by Bryan C. Mills's avatar Bryan C. Mills Committed by Bryan Mills

net/rpc: use a sync.Map for serviceMap instead of RWMutex

This has no measurable impact on performance, but somewhat simplifies
the code.

updates #18177

name                  old time/op    new time/op    delta
EndToEnd                54.3µs ±10%    55.7µs ±12%    ~     (p=0.505 n=8+8)
EndToEnd-6              31.4µs ± 9%    32.7µs ± 6%    ~     (p=0.130 n=8+8)
EndToEnd-48             25.5µs ±12%    26.4µs ± 6%    ~     (p=0.195 n=8+8)
EndToEndHTTP            53.7µs ± 8%    51.2µs ±15%    ~     (p=0.463 n=7+8)
EndToEndHTTP-6          30.9µs ±18%    31.2µs ±14%    ~     (p=0.959 n=8+8)
EndToEndHTTP-48         24.9µs ±11%    25.7µs ± 6%    ~     (p=0.382 n=8+8)
EndToEndAsync           23.6µs ± 7%    24.2µs ± 6%    ~     (p=0.383 n=7+7)
EndToEndAsync-6         21.0µs ±23%    22.0µs ±20%    ~     (p=0.574 n=8+8)
EndToEndAsync-48        22.8µs ±16%    23.3µs ±13%    ~     (p=0.721 n=8+8)
EndToEndAsyncHTTP       25.8µs ± 7%    24.7µs ±14%    ~     (p=0.161 n=8+8)
EndToEndAsyncHTTP-6     22.1µs ±19%    22.6µs ±12%    ~     (p=0.645 n=8+8)
EndToEndAsyncHTTP-48    22.9µs ±13%    22.1µs ±20%    ~     (p=0.574 n=8+8)

name                  old alloc/op   new alloc/op   delta
EndToEnd                  320B ± 0%      321B ± 0%    ~     (p=1.000 n=8+8)
EndToEnd-6                320B ± 0%      321B ± 0%  +0.20%  (p=0.037 n=8+7)
EndToEnd-48               326B ± 0%      326B ± 0%    ~     (p=0.124 n=8+8)
EndToEndHTTP              320B ± 0%      320B ± 0%    ~     (all equal)
EndToEndHTTP-6            320B ± 0%      321B ± 0%    ~     (p=0.077 n=8+8)
EndToEndHTTP-48           324B ± 0%      324B ± 0%    ~     (p=1.000 n=8+8)
EndToEndAsync             227B ± 0%      227B ± 0%    ~     (p=0.154 n=8+7)
EndToEndAsync-6           226B ± 0%      226B ± 0%    ~     (all equal)
EndToEndAsync-48          230B ± 1%      229B ± 1%    ~     (p=0.072 n=8+8)
EndToEndAsyncHTTP         227B ± 0%      227B ± 0%    ~     (all equal)
EndToEndAsyncHTTP-6       226B ± 0%      226B ± 0%    ~     (p=0.400 n=8+7)
EndToEndAsyncHTTP-48      228B ± 0%      228B ± 0%    ~     (p=0.949 n=8+6)

name                  old allocs/op  new allocs/op  delta
EndToEnd                  9.00 ± 0%      9.00 ± 0%    ~     (all equal)
EndToEnd-6                9.00 ± 0%      9.00 ± 0%    ~     (all equal)
EndToEnd-48               9.00 ± 0%      9.00 ± 0%    ~     (all equal)
EndToEndHTTP              9.00 ± 0%      9.00 ± 0%    ~     (all equal)
EndToEndHTTP-6            9.00 ± 0%      9.00 ± 0%    ~     (all equal)
EndToEndHTTP-48           9.00 ± 0%      9.00 ± 0%    ~     (all equal)
EndToEndAsync             8.00 ± 0%      8.00 ± 0%    ~     (all equal)
EndToEndAsync-6           8.00 ± 0%      8.00 ± 0%    ~     (all equal)
EndToEndAsync-48          8.00 ± 0%      8.00 ± 0%    ~     (all equal)
EndToEndAsyncHTTP         8.00 ± 0%      8.00 ± 0%    ~     (all equal)
EndToEndAsyncHTTP-6       8.00 ± 0%      8.00 ± 0%    ~     (all equal)
EndToEndAsyncHTTP-48      8.00 ± 0%      8.00 ± 0%    ~     (all equal)

https://perf.golang.org/search?q=upload:20170428.2

Change-Id: I8ef7f71a7602302aa78c144327270dfce9211539
Reviewed-on: https://go-review.googlesource.com/42112
Run-TryBot: Bryan Mills <bcmills@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent d6ce7e4f
...@@ -71,20 +71,17 @@ type debugHTTP struct { ...@@ -71,20 +71,17 @@ type debugHTTP struct {
// Runs at /debug/rpc // Runs at /debug/rpc
func (server debugHTTP) ServeHTTP(w http.ResponseWriter, req *http.Request) { func (server debugHTTP) ServeHTTP(w http.ResponseWriter, req *http.Request) {
// Build a sorted version of the data. // Build a sorted version of the data.
var services = make(serviceArray, len(server.serviceMap)) var services serviceArray
i := 0 server.serviceMap.Range(func(snamei, svci interface{}) bool {
server.mu.Lock() svc := svci.(*service)
for sname, service := range server.serviceMap { ds := debugService{svc, snamei.(string), make(methodArray, 0, len(svc.method))}
services[i] = debugService{service, sname, make(methodArray, len(service.method))} for mname, method := range svc.method {
j := 0 ds.Method = append(ds.Method, debugMethod{method, mname})
for mname, method := range service.method {
services[i].Method[j] = debugMethod{method, mname}
j++
} }
sort.Sort(services[i].Method) sort.Sort(ds.Method)
i++ services = append(services, ds)
} return true
server.mu.Unlock() })
sort.Sort(services) sort.Sort(services)
err := debug.Execute(w, services) err := debug.Execute(w, services)
if err != nil { if err != nil {
......
...@@ -187,8 +187,7 @@ type Response struct { ...@@ -187,8 +187,7 @@ type Response struct {
// Server represents an RPC Server. // Server represents an RPC Server.
type Server struct { type Server struct {
mu sync.RWMutex // protects the serviceMap serviceMap sync.Map // map[string]*service
serviceMap map[string]*service
reqLock sync.Mutex // protects freeReq reqLock sync.Mutex // protects freeReq
freeReq *Request freeReq *Request
respLock sync.Mutex // protects freeResp respLock sync.Mutex // protects freeResp
...@@ -197,7 +196,7 @@ type Server struct { ...@@ -197,7 +196,7 @@ type Server struct {
// NewServer returns a new Server. // NewServer returns a new Server.
func NewServer() *Server { func NewServer() *Server {
return &Server{serviceMap: make(map[string]*service)} return &Server{}
} }
// DefaultServer is the default instance of *Server. // DefaultServer is the default instance of *Server.
...@@ -240,11 +239,6 @@ func (server *Server) RegisterName(name string, rcvr interface{}) error { ...@@ -240,11 +239,6 @@ func (server *Server) RegisterName(name string, rcvr interface{}) error {
} }
func (server *Server) register(rcvr interface{}, name string, useName bool) error { func (server *Server) register(rcvr interface{}, name string, useName bool) error {
server.mu.Lock()
defer server.mu.Unlock()
if server.serviceMap == nil {
server.serviceMap = make(map[string]*service)
}
s := new(service) s := new(service)
s.typ = reflect.TypeOf(rcvr) s.typ = reflect.TypeOf(rcvr)
s.rcvr = reflect.ValueOf(rcvr) s.rcvr = reflect.ValueOf(rcvr)
...@@ -262,9 +256,6 @@ func (server *Server) register(rcvr interface{}, name string, useName bool) erro ...@@ -262,9 +256,6 @@ func (server *Server) register(rcvr interface{}, name string, useName bool) erro
log.Print(s) log.Print(s)
return errors.New(s) return errors.New(s)
} }
if _, present := server.serviceMap[sname]; present {
return errors.New("rpc: service already defined: " + sname)
}
s.name = sname s.name = sname
// Install the methods // Install the methods
...@@ -283,7 +274,10 @@ func (server *Server) register(rcvr interface{}, name string, useName bool) erro ...@@ -283,7 +274,10 @@ func (server *Server) register(rcvr interface{}, name string, useName bool) erro
log.Print(str) log.Print(str)
return errors.New(str) return errors.New(str)
} }
server.serviceMap[s.name] = s
if _, dup := server.serviceMap.LoadOrStore(sname, s); dup {
return errors.New("rpc: service already defined: " + sname)
}
return nil return nil
} }
...@@ -581,7 +575,7 @@ func (server *Server) readRequest(codec ServerCodec) (service *service, mtype *m ...@@ -581,7 +575,7 @@ func (server *Server) readRequest(codec ServerCodec) (service *service, mtype *m
return return
} }
func (server *Server) readRequestHeader(codec ServerCodec) (service *service, mtype *methodType, req *Request, keepReading bool, err error) { func (server *Server) readRequestHeader(codec ServerCodec) (svc *service, mtype *methodType, req *Request, keepReading bool, err error) {
// Grab the request header. // Grab the request header.
req = server.getRequest() req = server.getRequest()
err = codec.ReadRequestHeader(req) err = codec.ReadRequestHeader(req)
...@@ -607,14 +601,13 @@ func (server *Server) readRequestHeader(codec ServerCodec) (service *service, mt ...@@ -607,14 +601,13 @@ func (server *Server) readRequestHeader(codec ServerCodec) (service *service, mt
methodName := req.ServiceMethod[dot+1:] methodName := req.ServiceMethod[dot+1:]
// Look up the request. // Look up the request.
server.mu.RLock() svci, ok := server.serviceMap.Load(serviceName)
service = server.serviceMap[serviceName] if !ok {
server.mu.RUnlock()
if service == nil {
err = errors.New("rpc: can't find service " + req.ServiceMethod) err = errors.New("rpc: can't find service " + req.ServiceMethod)
return return
} }
mtype = service.method[methodName] svc = svci.(*service)
mtype = svc.method[methodName]
if mtype == nil { if mtype == nil {
err = errors.New("rpc: can't find method " + req.ServiceMethod) err = errors.New("rpc: can't find method " + req.ServiceMethod)
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment