| // Copyright 2014 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| // Goroutine-related profiles. |
| |
| package main |
| |
| import ( |
| "fmt" |
| "html/template" |
| "internal/trace" |
| "log" |
| "net/http" |
| "reflect" |
| "sort" |
| "strconv" |
| "sync" |
| "time" |
| ) |
| |
| func init() { |
| http.HandleFunc("/goroutines", httpGoroutines) |
| http.HandleFunc("/goroutine", httpGoroutine) |
| } |
| |
| // gtype describes a group of goroutines grouped by start PC. |
| type gtype struct { |
| ID uint64 // Unique identifier (PC). |
| Name string // Start function. |
| N int // Total number of goroutines in this group. |
| ExecTime int64 // Total execution time of all goroutines in this group. |
| } |
| |
| var ( |
| gsInit sync.Once |
| gs map[uint64]*trace.GDesc |
| ) |
| |
| // analyzeGoroutines generates statistics about execution of all goroutines and stores them in gs. |
| func analyzeGoroutines(events []*trace.Event) { |
| gsInit.Do(func() { |
| gs = trace.GoroutineStats(events) |
| }) |
| } |
| |
| // httpGoroutines serves list of goroutine groups. |
| func httpGoroutines(w http.ResponseWriter, r *http.Request) { |
| events, err := parseEvents() |
| if err != nil { |
| http.Error(w, err.Error(), http.StatusInternalServerError) |
| return |
| } |
| analyzeGoroutines(events) |
| gss := make(map[uint64]gtype) |
| for _, g := range gs { |
| gs1 := gss[g.PC] |
| gs1.ID = g.PC |
| gs1.Name = g.Name |
| gs1.N++ |
| gs1.ExecTime += g.ExecTime |
| gss[g.PC] = gs1 |
| } |
| var glist []gtype |
| for k, v := range gss { |
| v.ID = k |
| glist = append(glist, v) |
| } |
| sort.Slice(glist, func(i, j int) bool { return glist[i].ExecTime > glist[j].ExecTime }) |
| w.Header().Set("Content-Type", "text/html;charset=utf-8") |
| if err := templGoroutines.Execute(w, glist); err != nil { |
| log.Printf("failed to execute template: %v", err) |
| return |
| } |
| } |
| |
| var templGoroutines = template.Must(template.New("").Parse(` |
| <html> |
| <body> |
| Goroutines: <br> |
| {{range $}} |
| <a href="/goroutine?id={{.ID}}">{{.Name}}</a> N={{.N}} <br> |
| {{end}} |
| </body> |
| </html> |
| `)) |
| |
| // httpGoroutine serves list of goroutines in a particular group. |
| func httpGoroutine(w http.ResponseWriter, r *http.Request) { |
| // TODO(hyangah): support format=csv (raw data) |
| |
| events, err := parseEvents() |
| if err != nil { |
| http.Error(w, err.Error(), http.StatusInternalServerError) |
| return |
| } |
| |
| pc, err := strconv.ParseUint(r.FormValue("id"), 10, 64) |
| if err != nil { |
| http.Error(w, fmt.Sprintf("failed to parse id parameter '%v': %v", r.FormValue("id"), err), http.StatusInternalServerError) |
| return |
| } |
| analyzeGoroutines(events) |
| var ( |
| glist []*trace.GDesc |
| name string |
| totalExecTime, execTime int64 |
| maxTotalTime int64 |
| ) |
| |
| for _, g := range gs { |
| totalExecTime += g.ExecTime |
| |
| if g.PC != pc { |
| continue |
| } |
| glist = append(glist, g) |
| name = g.Name |
| execTime += g.ExecTime |
| if maxTotalTime < g.TotalTime { |
| maxTotalTime = g.TotalTime |
| } |
| } |
| |
| execTimePercent := "" |
| if totalExecTime > 0 { |
| execTimePercent = fmt.Sprintf("%.2f%%", float64(execTime)/float64(totalExecTime)*100) |
| } |
| |
| sortby := r.FormValue("sortby") |
| _, ok := reflect.TypeOf(trace.GDesc{}).FieldByNameFunc(func(s string) bool { |
| return s == sortby |
| }) |
| if !ok { |
| sortby = "TotalTime" |
| } |
| |
| sort.Slice(glist, func(i, j int) bool { |
| ival := reflect.ValueOf(glist[i]).Elem().FieldByName(sortby).Int() |
| jval := reflect.ValueOf(glist[j]).Elem().FieldByName(sortby).Int() |
| return ival > jval |
| }) |
| |
| err = templGoroutine.Execute(w, struct { |
| Name string |
| PC uint64 |
| N int |
| ExecTimePercent string |
| MaxTotal int64 |
| GList []*trace.GDesc |
| }{ |
| Name: name, |
| PC: pc, |
| N: len(glist), |
| ExecTimePercent: execTimePercent, |
| MaxTotal: maxTotalTime, |
| GList: glist}) |
| if err != nil { |
| http.Error(w, fmt.Sprintf("failed to execute template: %v", err), http.StatusInternalServerError) |
| return |
| } |
| } |
| |
| var templGoroutine = template.Must(template.New("").Funcs(template.FuncMap{ |
| "prettyDuration": func(nsec int64) template.HTML { |
| d := time.Duration(nsec) * time.Nanosecond |
| return template.HTML(niceDuration(d)) |
| }, |
| "percent": func(dividend, divisor int64) template.HTML { |
| if divisor == 0 { |
| return "" |
| } |
| return template.HTML(fmt.Sprintf("(%.1f%%)", float64(dividend)/float64(divisor)*100)) |
| }, |
| "barLen": func(dividend, divisor int64) template.HTML { |
| if divisor == 0 { |
| return "0" |
| } |
| return template.HTML(fmt.Sprintf("%.2f%%", float64(dividend)/float64(divisor)*100)) |
| }, |
| "unknownTime": func(desc *trace.GDesc) int64 { |
| sum := desc.ExecTime + desc.IOTime + desc.BlockTime + desc.SyscallTime + desc.SchedWaitTime |
| if sum < desc.TotalTime { |
| return desc.TotalTime - sum |
| } |
| return 0 |
| }, |
| }).Parse(` |
| <!DOCTYPE html> |
| <title>Goroutine {{.Name}}</title> |
| <style> |
| th { |
| background-color: #050505; |
| color: #fff; |
| } |
| table { |
| border-collapse: collapse; |
| } |
| .details tr:hover { |
| background-color: #f2f2f2; |
| } |
| .details td { |
| text-align: right; |
| border: 1px solid black; |
| } |
| .details td.id { |
| text-align: left; |
| } |
| .stacked-bar-graph { |
| width: 300px; |
| height: 10px; |
| color: #414042; |
| white-space: nowrap; |
| font-size: 5px; |
| } |
| .stacked-bar-graph span { |
| display: inline-block; |
| width: 100%; |
| height: 100%; |
| box-sizing: border-box; |
| float: left; |
| padding: 0; |
| } |
| .unknown-time { background-color: #636363; } |
| .exec-time { background-color: #d7191c; } |
| .io-time { background-color: #fdae61; } |
| .block-time { background-color: #d01c8b; } |
| .syscall-time { background-color: #7b3294; } |
| .sched-time { background-color: #2c7bb6; } |
| </style> |
| |
| <script> |
| function reloadTable(key, value) { |
| let params = new URLSearchParams(window.location.search); |
| params.set(key, value); |
| window.location.search = params.toString(); |
| } |
| </script> |
| |
| <table class="summary"> |
| <tr><td>Goroutine Name:</td><td>{{.Name}}</td></tr> |
| <tr><td>Number of Goroutines:</td><td>{{.N}}</td></tr> |
| <tr><td>Execution Time:</td><td>{{.ExecTimePercent}} of total program execution time </td> </tr> |
| <tr><td>Network Wait Time:</td><td> <a href="/io?id={{.PC}}">graph</a><a href="/io?id={{.PC}}&raw=1" download="io.profile">(download)</a></td></tr> |
| <tr><td>Sync Block Time:</td><td> <a href="/block?id={{.PC}}">graph</a><a href="/block?id={{.PC}}&raw=1" download="block.profile">(download)</a></td></tr> |
| <tr><td>Blocking Syscall Time:</td><td> <a href="/syscall?id={{.PC}}">graph</a><a href="/syscall?id={{.PC}}&raw=1" download="syscall.profile">(download)</a></td></tr> |
| <tr><td>Scheduler Wait Time:</td><td> <a href="/sched?id={{.PC}}">graph</a><a href="/sched?id={{.PC}}&raw=1" download="sched.profile">(download)</a></td></tr> |
| </table> |
| <p> |
| <table class="details"> |
| <tr> |
| <th> Goroutine</th> |
| <th onclick="reloadTable('sortby', 'TotalTime')"> Total</th> |
| <th></th> |
| <th onclick="reloadTable('sortby', 'ExecTime')" class="exec-time"> Execution</th> |
| <th onclick="reloadTable('sortby', 'IOTime')" class="io-time"> Network wait</th> |
| <th onclick="reloadTable('sortby', 'BlockTime')" class="block-time"> Sync block </th> |
| <th onclick="reloadTable('sortby', 'SyscallTime')" class="syscall-time"> Blocking syscall</th> |
| <th onclick="reloadTable('sortby', 'SchedWaitTime')" class="sched-time"> Scheduler wait</th> |
| <th onclick="reloadTable('sortby', 'SweepTime')"> GC sweeping</th> |
| <th onclick="reloadTable('sortby', 'GCTime')"> GC pause</th> |
| </tr> |
| {{range .GList}} |
| <tr> |
| <td> <a href="/trace?goid={{.ID}}">{{.ID}}</a> </td> |
| <td> {{prettyDuration .TotalTime}} </td> |
| <td> |
| <div class="stacked-bar-graph"> |
| {{if unknownTime .}}<span style="width:{{barLen (unknownTime .) $.MaxTotal}}" class="unknown-time"> </span>{{end}} |
| {{if .ExecTime}}<span style="width:{{barLen .ExecTime $.MaxTotal}}" class="exec-time"> </span>{{end}} |
| {{if .IOTime}}<span style="width:{{barLen .IOTime $.MaxTotal}}" class="io-time"> </span>{{end}} |
| {{if .BlockTime}}<span style="width:{{barLen .BlockTime $.MaxTotal}}" class="block-time"> </span>{{end}} |
| {{if .SyscallTime}}<span style="width:{{barLen .SyscallTime $.MaxTotal}}" class="syscall-time"> </span>{{end}} |
| {{if .SchedWaitTime}}<span style="width:{{barLen .SchedWaitTime $.MaxTotal}}" class="sched-time"> </span>{{end}} |
| </div> |
| </td> |
| <td> {{prettyDuration .ExecTime}}</td> |
| <td> {{prettyDuration .IOTime}}</td> |
| <td> {{prettyDuration .BlockTime}}</td> |
| <td> {{prettyDuration .SyscallTime}}</td> |
| <td> {{prettyDuration .SchedWaitTime}}</td> |
| <td> {{prettyDuration .SweepTime}} {{percent .SweepTime .TotalTime}}</td> |
| <td> {{prettyDuration .GCTime}} {{percent .GCTime .TotalTime}}</td> |
| </tr> |
| {{end}} |
| </table> |
| `)) |