| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | package driver |
| |
|
| | import ( |
| | "errors" |
| | "fmt" |
| | "os" |
| |
|
| | "github.com/google/pprof/internal/binutils" |
| | "github.com/google/pprof/internal/plugin" |
| | ) |
| |
|
| | type source struct { |
| | Sources []string |
| | ExecName string |
| | BuildID string |
| | Base []string |
| | DiffBase bool |
| | Normalize bool |
| |
|
| | Seconds int |
| | Timeout int |
| | Symbolize string |
| | HTTPHostport string |
| | HTTPDisableBrowser bool |
| | Comment string |
| | } |
| |
|
| | |
| | |
| | |
| | func parseFlags(o *plugin.Options) (*source, []string, error) { |
| | flag := o.Flagset |
| | |
| | flagDiffBase := flag.StringList("diff_base", "", "Source of base profile for comparison") |
| | flagBase := flag.StringList("base", "", "Source of base profile for profile subtraction") |
| | |
| | flagSymbolize := flag.String("symbolize", "", "Options for profile symbolization") |
| | flagBuildID := flag.String("buildid", "", "Override build id for first mapping") |
| | flagTimeout := flag.Int("timeout", -1, "Timeout in seconds for fetching a profile") |
| | flagAddComment := flag.String("add_comment", "", "Annotation string to record in the profile") |
| | |
| | flagSeconds := flag.Int("seconds", -1, "Length of time for dynamic profiles") |
| | |
| | flagInUseSpace := flag.Bool("inuse_space", false, "Display in-use memory size") |
| | flagInUseObjects := flag.Bool("inuse_objects", false, "Display in-use object counts") |
| | flagAllocSpace := flag.Bool("alloc_space", false, "Display allocated memory size") |
| | flagAllocObjects := flag.Bool("alloc_objects", false, "Display allocated object counts") |
| | |
| | flagTotalDelay := flag.Bool("total_delay", false, "Display total delay at each region") |
| | flagContentions := flag.Bool("contentions", false, "Display number of delays at each region") |
| | flagMeanDelay := flag.Bool("mean_delay", false, "Display mean delay at each region") |
| | flagTools := flag.String("tools", os.Getenv("PPROF_TOOLS"), "Path for object tool pathnames") |
| |
|
| | flagHTTP := flag.String("http", "", "Present interactive web UI at the specified http host:port") |
| | flagNoBrowser := flag.Bool("no_browser", false, "Skip opening a browser for the interactive web UI") |
| |
|
| | |
| | cfg := currentConfig() |
| | configFlagSetter := installConfigFlags(flag, &cfg) |
| |
|
| | flagCommands := make(map[string]*bool) |
| | flagParamCommands := make(map[string]*string) |
| | for name, cmd := range pprofCommands { |
| | if cmd.hasParam { |
| | flagParamCommands[name] = flag.String(name, "", "Generate a report in "+name+" format, matching regexp") |
| | } else { |
| | flagCommands[name] = flag.Bool(name, false, "Generate a report in "+name+" format") |
| | } |
| | } |
| |
|
| | args := flag.Parse(func() { |
| | o.UI.Print(usageMsgHdr + |
| | usage(true) + |
| | usageMsgSrc + |
| | flag.ExtraUsage() + |
| | usageMsgVars) |
| | }) |
| | if len(args) == 0 { |
| | return nil, nil, errors.New("no profile source specified") |
| | } |
| |
|
| | var execName string |
| | |
| | if len(args) > 1 { |
| | arg0 := args[0] |
| | if file, err := o.Obj.Open(arg0, 0, ^uint64(0), 0, ""); err == nil { |
| | file.Close() |
| | execName = arg0 |
| | args = args[1:] |
| | } |
| | } |
| |
|
| | |
| | if err := configFlagSetter(); err != nil { |
| | return nil, nil, err |
| | } |
| |
|
| | cmd, err := outputFormat(flagCommands, flagParamCommands) |
| | if err != nil { |
| | return nil, nil, err |
| | } |
| | if cmd != nil && *flagHTTP != "" { |
| | return nil, nil, errors.New("-http is not compatible with an output format on the command line") |
| | } |
| |
|
| | if *flagNoBrowser && *flagHTTP == "" { |
| | return nil, nil, errors.New("-no_browser only makes sense with -http") |
| | } |
| |
|
| | si := cfg.SampleIndex |
| | si = sampleIndex(flagTotalDelay, si, "delay", "-total_delay", o.UI) |
| | si = sampleIndex(flagMeanDelay, si, "delay", "-mean_delay", o.UI) |
| | si = sampleIndex(flagContentions, si, "contentions", "-contentions", o.UI) |
| | si = sampleIndex(flagInUseSpace, si, "inuse_space", "-inuse_space", o.UI) |
| | si = sampleIndex(flagInUseObjects, si, "inuse_objects", "-inuse_objects", o.UI) |
| | si = sampleIndex(flagAllocSpace, si, "alloc_space", "-alloc_space", o.UI) |
| | si = sampleIndex(flagAllocObjects, si, "alloc_objects", "-alloc_objects", o.UI) |
| | cfg.SampleIndex = si |
| |
|
| | if *flagMeanDelay { |
| | cfg.Mean = true |
| | } |
| |
|
| | source := &source{ |
| | Sources: args, |
| | ExecName: execName, |
| | BuildID: *flagBuildID, |
| | Seconds: *flagSeconds, |
| | Timeout: *flagTimeout, |
| | Symbolize: *flagSymbolize, |
| | HTTPHostport: *flagHTTP, |
| | HTTPDisableBrowser: *flagNoBrowser, |
| | Comment: *flagAddComment, |
| | } |
| |
|
| | if err := source.addBaseProfiles(*flagBase, *flagDiffBase); err != nil { |
| | return nil, nil, err |
| | } |
| |
|
| | normalize := cfg.Normalize |
| | if normalize && len(source.Base) == 0 { |
| | return nil, nil, errors.New("must have base profile to normalize by") |
| | } |
| | source.Normalize = normalize |
| |
|
| | if bu, ok := o.Obj.(*binutils.Binutils); ok { |
| | bu.SetTools(*flagTools) |
| | } |
| |
|
| | setCurrentConfig(cfg) |
| | return source, cmd, nil |
| | } |
| |
|
| | |
| | |
| | |
| | func (source *source) addBaseProfiles(flagBase, flagDiffBase []*string) error { |
| | base, diffBase := dropEmpty(flagBase), dropEmpty(flagDiffBase) |
| | if len(base) > 0 && len(diffBase) > 0 { |
| | return errors.New("-base and -diff_base flags cannot both be specified") |
| | } |
| |
|
| | source.Base = base |
| | if len(diffBase) > 0 { |
| | source.Base, source.DiffBase = diffBase, true |
| | } |
| | return nil |
| | } |
| |
|
| | |
| | |
| | func dropEmpty(list []*string) []string { |
| | var l []string |
| | for _, s := range list { |
| | if *s != "" { |
| | l = append(l, *s) |
| | } |
| | } |
| | return l |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | func installConfigFlags(flag plugin.FlagSet, cfg *config) func() error { |
| | |
| | var setters []func() |
| | var err error |
| |
|
| | for _, field := range configFields { |
| | n := field.name |
| | help := configHelp[n] |
| | var setter func() |
| | switch ptr := cfg.fieldPtr(field).(type) { |
| | case *bool: |
| | f := flag.Bool(n, *ptr, help) |
| | setter = func() { *ptr = *f } |
| | case *int: |
| | f := flag.Int(n, *ptr, help) |
| | setter = func() { *ptr = *f } |
| | case *float64: |
| | f := flag.Float64(n, *ptr, help) |
| | setter = func() { *ptr = *f } |
| | case *string: |
| | if len(field.choices) == 0 { |
| | f := flag.String(n, *ptr, help) |
| | setter = func() { *ptr = *f } |
| | } else { |
| | |
| | |
| | |
| | bools := make(map[string]*bool) |
| | for _, choice := range field.choices { |
| | bools[choice] = flag.Bool(choice, false, configHelp[choice]) |
| | } |
| | setter = func() { |
| | var set []string |
| | for k, v := range bools { |
| | if *v { |
| | set = append(set, k) |
| | } |
| | } |
| | switch len(set) { |
| | case 0: |
| | |
| | case 1: |
| | *ptr = set[0] |
| | default: |
| | err = fmt.Errorf("conflicting options set: %v", set) |
| | } |
| | } |
| | } |
| | } |
| | setters = append(setters, setter) |
| | } |
| |
|
| | return func() error { |
| | |
| | for _, setter := range setters { |
| | setter() |
| | if err != nil { |
| | return err |
| | } |
| | } |
| | return nil |
| | } |
| | } |
| |
|
| | func sampleIndex(flag *bool, si string, sampleType, option string, ui plugin.UI) string { |
| | if *flag { |
| | if si == "" { |
| | return sampleType |
| | } |
| | ui.PrintErr("Multiple value selections, ignoring ", option) |
| | } |
| | return si |
| | } |
| |
|
| | func outputFormat(bcmd map[string]*bool, acmd map[string]*string) (cmd []string, err error) { |
| | for n, b := range bcmd { |
| | if *b { |
| | if cmd != nil { |
| | return nil, errors.New("must set at most one output format") |
| | } |
| | cmd = []string{n} |
| | } |
| | } |
| | for n, s := range acmd { |
| | if *s != "" { |
| | if cmd != nil { |
| | return nil, errors.New("must set at most one output format") |
| | } |
| | cmd = []string{n, *s} |
| | } |
| | } |
| | return cmd, nil |
| | } |
| |
|
| | var usageMsgHdr = `usage: |
| | |
| | Produce output in the specified format. |
| | |
| | pprof <format> [options] [binary] <source> ... |
| | |
| | Omit the format to get an interactive shell whose commands can be used |
| | to generate various views of a profile |
| | |
| | pprof [options] [binary] <source> ... |
| | |
| | Omit the format and provide the "-http" flag to get an interactive web |
| | interface at the specified host:port that can be used to navigate through |
| | various views of a profile. |
| | |
| | pprof -http [host]:[port] [options] [binary] <source> ... |
| | |
| | Details: |
| | ` |
| |
|
| | var usageMsgSrc = "\n\n" + |
| | " Source options:\n" + |
| | " -seconds Duration for time-based profile collection\n" + |
| | " -timeout Timeout in seconds for profile collection\n" + |
| | " -buildid Override build id for main binary\n" + |
| | " -add_comment Free-form annotation to add to the profile\n" + |
| | " Displayed on some reports or with pprof -comments\n" + |
| | " -diff_base source Source of base profile for comparison\n" + |
| | " -base source Source of base profile for profile subtraction\n" + |
| | " profile.pb.gz Profile in compressed protobuf format\n" + |
| | " legacy_profile Profile in legacy pprof format\n" + |
| | " http://host/profile URL for profile handler to retrieve\n" + |
| | " -symbolize= Controls source of symbol information\n" + |
| | " none Do not attempt symbolization\n" + |
| | " local Examine only local binaries\n" + |
| | " fastlocal Only get function names from local binaries\n" + |
| | " remote Do not examine local binaries\n" + |
| | " force Force re-symbolization\n" + |
| | " Binary Local path or build id of binary for symbolization\n" |
| |
|
| | var usageMsgVars = "\n\n" + |
| | " Misc options:\n" + |
| | " -http Provide web interface at host:port.\n" + |
| | " Host is optional and 'localhost' by default.\n" + |
| | " Port is optional and a randomly available port by default.\n" + |
| | " -no_browser Skip opening a browser for the interactive web UI.\n" + |
| | " -tools Search path for object tools\n" + |
| | "\n" + |
| | " Legacy convenience options:\n" + |
| | " -inuse_space Same as -sample_index=inuse_space\n" + |
| | " -inuse_objects Same as -sample_index=inuse_objects\n" + |
| | " -alloc_space Same as -sample_index=alloc_space\n" + |
| | " -alloc_objects Same as -sample_index=alloc_objects\n" + |
| | " -total_delay Same as -sample_index=delay\n" + |
| | " -contentions Same as -sample_index=contentions\n" + |
| | " -mean_delay Same as -mean -sample_index=delay\n" + |
| | "\n" + |
| | " Environment Variables:\n" + |
| | " PPROF_TMPDIR Location for saved profiles (default $HOME/pprof)\n" + |
| | " PPROF_TOOLS Search path for object-level tools\n" + |
| | " PPROF_BINARY_PATH Search path for local binary files\n" + |
| | " default: $HOME/pprof/binaries\n" + |
| | " searches $buildid/$name, $buildid/*, $path/$buildid,\n" + |
| | " ${buildid:0:2}/${buildid:2}.debug, $name, $path,\n" + |
| | " ${name}.debug, $dir/.debug/${name}.debug,\n" + |
| | " usr/lib/debug/$dir/${name}.debug\n" + |
| | " * On Windows, %USERPROFILE% is used instead of $HOME" |
| |
|