123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106 |
- // Copyright 2017 Google Inc. All Rights Reserved.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
-
- package driver
-
- import (
- "encoding/json"
- "html/template"
- "net/http"
- "strings"
-
- "github.com/google/pprof/internal/graph"
- "github.com/google/pprof/internal/measurement"
- "github.com/google/pprof/internal/report"
- )
-
- type treeNode struct {
- Name string `json:"n"`
- FullName string `json:"f"`
- Cum int64 `json:"v"`
- CumFormat string `json:"l"`
- Percent string `json:"p"`
- Children []*treeNode `json:"c"`
- }
-
- // flamegraph generates a web page containing a flamegraph.
- func (ui *webInterface) flamegraph(w http.ResponseWriter, req *http.Request) {
- // Force the call tree so that the graph is a tree.
- // Also do not trim the tree so that the flame graph contains all functions.
- rpt, errList := ui.makeReport(w, req, []string{"svg"}, func(cfg *config) {
- cfg.CallTree = true
- cfg.Trim = false
- })
- if rpt == nil {
- return // error already reported
- }
-
- // Generate dot graph.
- g, config := report.GetDOT(rpt)
- var nodes []*treeNode
- nroots := 0
- rootValue := int64(0)
- nodeArr := []string{}
- nodeMap := map[*graph.Node]*treeNode{}
- // Make all nodes and the map, collect the roots.
- for _, n := range g.Nodes {
- v := n.CumValue()
- fullName := n.Info.PrintableName()
- node := &treeNode{
- Name: graph.ShortenFunctionName(fullName),
- FullName: fullName,
- Cum: v,
- CumFormat: config.FormatValue(v),
- Percent: strings.TrimSpace(measurement.Percentage(v, config.Total)),
- }
- nodes = append(nodes, node)
- if len(n.In) == 0 {
- nodes[nroots], nodes[len(nodes)-1] = nodes[len(nodes)-1], nodes[nroots]
- nroots++
- rootValue += v
- }
- nodeMap[n] = node
- // Get all node names into an array.
- nodeArr = append(nodeArr, n.Info.Name)
- }
- // Populate the child links.
- for _, n := range g.Nodes {
- node := nodeMap[n]
- for child := range n.Out {
- node.Children = append(node.Children, nodeMap[child])
- }
- }
-
- rootNode := &treeNode{
- Name: "root",
- FullName: "root",
- Cum: rootValue,
- CumFormat: config.FormatValue(rootValue),
- Percent: strings.TrimSpace(measurement.Percentage(rootValue, config.Total)),
- Children: nodes[0:nroots],
- }
-
- // JSON marshalling flame graph
- b, err := json.Marshal(rootNode)
- if err != nil {
- http.Error(w, "error serializing flame graph", http.StatusInternalServerError)
- ui.options.UI.PrintErr(err)
- return
- }
-
- ui.render(w, req, "flamegraph", rpt, errList, config.Labels, webArgs{
- FlameGraph: template.JS(b),
- Nodes: nodeArr,
- })
- }
|