Add options to control the order of profile output.
This commit is contained in:
parent
81bb14ebf1
commit
109ee25287
|
@ -23,6 +23,7 @@ function that consumes a @racket[profile] instance. See the
|
||||||
|
|
||||||
@defproc[(render
|
@defproc[(render
|
||||||
[profile-data profile?]
|
[profile-data profile?]
|
||||||
|
[order (or/c 'topological 'self 'total) 'topological]
|
||||||
[#:truncate-source truncate-source exact-nonnegative-integer? 50]
|
[#:truncate-source truncate-source exact-nonnegative-integer? 50]
|
||||||
[#:hide-self hide-self% (between/c 0 1) 1/100]
|
[#:hide-self hide-self% (between/c 0 1) 1/100]
|
||||||
[#:hide-subs hide-subs% (between/c 0 1) 2/100])
|
[#:hide-subs hide-subs% (between/c 0 1) 2/100])
|
||||||
|
@ -32,7 +33,7 @@ Prints the given @racket[profile] results as a textual table.
|
||||||
|
|
||||||
The printout begins with general information about the profile,
|
The printout begins with general information about the profile,
|
||||||
followed by a table with an entry for each node in the call graph.
|
followed by a table with an entry for each node in the call graph.
|
||||||
The entries are displayed in a topological order (roughly, since the
|
The entries are displayed in a topological order by default (roughly, since the
|
||||||
graph can have cycles). This means that it is usually easy to find
|
graph can have cycles). This means that it is usually easy to find
|
||||||
the callers and callees of a function in its close environment.
|
the callers and callees of a function in its close environment.
|
||||||
|
|
||||||
|
@ -105,6 +106,13 @@ The function has a few keyword arguments to customize its output:
|
||||||
conditions is to avoid having ``dangling references'' to hidden
|
conditions is to avoid having ``dangling references'' to hidden
|
||||||
nodes.}
|
nodes.}
|
||||||
|
|
||||||
|
@item{The @racket[order] argument determines the order in which entries
|
||||||
|
appear in the output. If @racket[order] is @racket['topological] (the default),
|
||||||
|
entries are sorted topologically, grouping callers and callees close together.
|
||||||
|
If @racket[order] is @racket['self], entries are sorted by how often
|
||||||
|
they appear at the top of a stack snapshot. If @racket[order] is @racket['total],
|
||||||
|
entries are sorted by how often they appear anywhere in a stack snapshot.}
|
||||||
|
|
||||||
]}
|
]}
|
||||||
|
|
||||||
|
|
||||||
|
@ -115,6 +123,7 @@ The function has a few keyword arguments to customize its output:
|
||||||
|
|
||||||
@defproc[(render
|
@defproc[(render
|
||||||
[profile-data profile?]
|
[profile-data profile?]
|
||||||
|
[order (or/c 'topological 'self 'total) 'topological]
|
||||||
[#:hide-self hide-self% (between/c 0 1) 1/100]
|
[#:hide-self hide-self% (between/c 0 1) 1/100]
|
||||||
[#:hide-subs hide-subs% (between/c 0 1) 2/100])
|
[#:hide-subs hide-subs% (between/c 0 1) 2/100])
|
||||||
void?]{
|
void?]{
|
||||||
|
@ -128,4 +137,4 @@ of the Graphviz tools to render. Nodes are colored according to their
|
||||||
`self' percentages, and edges.
|
`self' percentages, and edges.
|
||||||
|
|
||||||
The keyword arguments control hiding nodes in the same way as with the
|
The keyword arguments control hiding nodes in the same way as with the
|
||||||
textual renderer.}
|
textual renderer. The @racket[order] argument is ignored.}
|
||||||
|
|
|
@ -30,12 +30,15 @@ intended as a convenient tool for profiling code.
|
||||||
[#:delay delay (>=/c 0.0) 0.05]
|
[#:delay delay (>=/c 0.0) 0.05]
|
||||||
[#:repeat iterations exact-nonnegative-integer? 1]
|
[#:repeat iterations exact-nonnegative-integer? 1]
|
||||||
[#:threads threads? any/c #f]
|
[#:threads threads? any/c #f]
|
||||||
[#:render renderer (profile? . -> . any/c) text:render]
|
[#:render renderer (-> profile? (or/c 'topological 'self 'total) any/c) text:render]
|
||||||
[#:periodic-renderer periodic-renderer
|
[#:periodic-renderer periodic-renderer
|
||||||
(or/c #f (list/c (>=/c 0.0)
|
(or/c #f (list/c (>=/c 0.0)
|
||||||
(profile? . -> . any/c)))
|
(-> profile?
|
||||||
|
(or/c 'topological 'self 'total)
|
||||||
|
any/c)))
|
||||||
#f]
|
#f]
|
||||||
[#:use-errortrace? use-errortrace? any/c #f])
|
[#:use-errortrace? use-errortrace? any/c #f]
|
||||||
|
[#:order order (or/c 'topological 'self 'total) 'topological])
|
||||||
any/c]{
|
any/c]{
|
||||||
|
|
||||||
Executes the given @racket[thunk] and collect profiling data during
|
Executes the given @racket[thunk] and collect profiling data during
|
||||||
|
@ -88,6 +91,12 @@ Keyword arguments can customize the profiling:
|
||||||
using @racket[errortrace-compile-handler], and the profiled program must be
|
using @racket[errortrace-compile-handler], and the profiled program must be
|
||||||
run using @commandline{racket -l errortrace -t program.rkt} Removing compiled
|
run using @commandline{racket -l errortrace -t program.rkt} Removing compiled
|
||||||
files (with extension @tt{.zo}) is sufficient to enable this.}
|
files (with extension @tt{.zo}) is sufficient to enable this.}
|
||||||
|
|
||||||
|
@item{The @racket[order] value is passed to the @racket[renderer] to control the
|
||||||
|
order of its output. By default, entries in the profile are sorted
|
||||||
|
topologically, but they can also be sorted by the time an entry is on top of
|
||||||
|
the stack (@racket['self]) or appears anywhere on the stack (@racket['total]).
|
||||||
|
Some renderers may ignore this option.}
|
||||||
]}
|
]}
|
||||||
|
|
||||||
@defform[(profile expr keyword-arguments ...)]{
|
@defform[(profile expr keyword-arguments ...)]{
|
||||||
|
|
|
@ -12,7 +12,15 @@
|
||||||
#:threads [threads? #f]
|
#:threads [threads? #f]
|
||||||
#:render [renderer text:render]
|
#:render [renderer text:render]
|
||||||
#:periodic-renderer [periodic-renderer #f]
|
#:periodic-renderer [periodic-renderer #f]
|
||||||
#:use-errortrace? [et? #f])
|
#:use-errortrace? [et? #f]
|
||||||
|
#:order [order 'topological])
|
||||||
|
(unless (member order '(topological self total))
|
||||||
|
(raise-argument-error
|
||||||
|
'profile-thunk "(or/c 'topological 'self 'total)" order))
|
||||||
|
(define (call-renderer renderer profile)
|
||||||
|
(if (procedure-arity-includes? renderer 2)
|
||||||
|
(renderer profile order)
|
||||||
|
(renderer profile)))
|
||||||
(define cust (and threads? (make-custodian (current-custodian))))
|
(define cust (and threads? (make-custodian (current-custodian))))
|
||||||
(define sampler (create-sampler (if threads?
|
(define sampler (create-sampler (if threads?
|
||||||
(list cust (current-thread))
|
(list cust (current-thread))
|
||||||
|
@ -25,7 +33,7 @@
|
||||||
[renderer (cadr periodic-renderer)])
|
[renderer (cadr periodic-renderer)])
|
||||||
(define (loop)
|
(define (loop)
|
||||||
(sleep delay)
|
(sleep delay)
|
||||||
(renderer (analyze-samples (sampler 'get-snapshots)))
|
(call-renderer renderer (analyze-samples (sampler 'get-snapshots)))
|
||||||
(loop))
|
(loop))
|
||||||
(thread loop))))
|
(thread loop))))
|
||||||
(define (run) (for/last ([i (in-range rpt)]) (thunk)))
|
(define (run) (for/last ([i (in-range rpt)]) (thunk)))
|
||||||
|
@ -38,7 +46,7 @@
|
||||||
(run)))
|
(run)))
|
||||||
(when periodic-thread (kill-thread periodic-thread))
|
(when periodic-thread (kill-thread periodic-thread))
|
||||||
(sampler 'stop)
|
(sampler 'stop)
|
||||||
(renderer (analyze-samples (sampler 'get-snapshots)))))
|
(call-renderer renderer (analyze-samples (sampler 'get-snapshots)))))
|
||||||
|
|
||||||
(define-syntax (profile stx)
|
(define-syntax (profile stx)
|
||||||
(syntax-case stx ()
|
(syntax-case stx ()
|
||||||
|
|
|
@ -3,7 +3,8 @@
|
||||||
(require racket/cmdline
|
(require racket/cmdline
|
||||||
raco/command-name
|
raco/command-name
|
||||||
errortrace/errortrace-lib
|
errortrace/errortrace-lib
|
||||||
"main.rkt" "raco-utils.rkt")
|
"main.rkt" "raco-utils.rkt"
|
||||||
|
(prefix-in text: "render-text.rkt"))
|
||||||
|
|
||||||
;; raco profile
|
;; raco profile
|
||||||
;; profile the main submodule (if there is one), or the top-level module
|
;; profile the main submodule (if there is one), or the top-level module
|
||||||
|
@ -12,6 +13,7 @@
|
||||||
(define iterations #f)
|
(define iterations #f)
|
||||||
(define threads? #f)
|
(define threads? #f)
|
||||||
(define use-errortrace? #f)
|
(define use-errortrace? #f)
|
||||||
|
(define order 'topological)
|
||||||
(define file
|
(define file
|
||||||
(command-line #:program (short-program+command-name)
|
(command-line #:program (short-program+command-name)
|
||||||
#:once-each
|
#:once-each
|
||||||
|
@ -33,6 +35,16 @@
|
||||||
[("--use-errortrace")
|
[("--use-errortrace")
|
||||||
"Use errortrace mode"
|
"Use errortrace mode"
|
||||||
(set! use-errortrace? #t)]
|
(set! use-errortrace? #t)]
|
||||||
|
#:once-any
|
||||||
|
[("--topological")
|
||||||
|
"Order functions topologically (the default)"
|
||||||
|
(set! order 'topological)]
|
||||||
|
[("--self")
|
||||||
|
"Order functions by self time"
|
||||||
|
(set! order 'self)]
|
||||||
|
[("--total")
|
||||||
|
"Order functions by total time"
|
||||||
|
(set! order 'total)]
|
||||||
#:args (filename)
|
#:args (filename)
|
||||||
filename))
|
filename))
|
||||||
|
|
||||||
|
@ -49,20 +61,24 @@
|
||||||
(profile-thunk t
|
(profile-thunk t
|
||||||
#:delay delay
|
#:delay delay
|
||||||
#:repeat iterations
|
#:repeat iterations
|
||||||
|
#:order order
|
||||||
#:threads threads?
|
#:threads threads?
|
||||||
#:use-errortrace? use-errortrace?)]
|
#:use-errortrace? use-errortrace?)]
|
||||||
[delay
|
[delay
|
||||||
(profile-thunk t
|
(profile-thunk t
|
||||||
#:delay delay
|
#:delay delay
|
||||||
|
#:order order
|
||||||
#:threads threads?
|
#:threads threads?
|
||||||
#:use-errortrace? use-errortrace?)]
|
#:use-errortrace? use-errortrace?)]
|
||||||
[iterations
|
[iterations
|
||||||
(profile-thunk t
|
(profile-thunk t
|
||||||
#:repeat iterations
|
#:repeat iterations
|
||||||
|
#:order order
|
||||||
#:threads threads?
|
#:threads threads?
|
||||||
#:use-errortrace? use-errortrace?)]
|
#:use-errortrace? use-errortrace?)]
|
||||||
[else
|
[else
|
||||||
(profile-thunk t
|
(profile-thunk t
|
||||||
|
#:order order
|
||||||
#:threads threads?
|
#:threads threads?
|
||||||
#:use-errortrace? use-errortrace?)]))
|
#:use-errortrace? use-errortrace?)]))
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
(require "analyzer.rkt" "utils.rkt")
|
(require "analyzer.rkt" "utils.rkt")
|
||||||
|
|
||||||
(define (render profile
|
(define (render profile
|
||||||
|
[order 'topological]
|
||||||
#:hide-self [hide-self% 1/100]
|
#:hide-self [hide-self% 1/100]
|
||||||
#:hide-subs [hide-subs% 2/100])
|
#:hide-subs [hide-subs% 2/100])
|
||||||
(define *-node (profile-*-node profile))
|
(define *-node (profile-*-node profile))
|
||||||
|
|
|
@ -41,9 +41,13 @@
|
||||||
[else (error 'internal-error "poof")])))))
|
[else (error 'internal-error "poof")])))))
|
||||||
|
|
||||||
(define (render profile
|
(define (render profile
|
||||||
|
[order 'topological]
|
||||||
#:truncate-source [truncate-source 50]
|
#:truncate-source [truncate-source 50]
|
||||||
#:hide-self [hide-self% 1/100]
|
#:hide-self [hide-self% 1/100]
|
||||||
#:hide-subs [hide-subs% 2/100])
|
#:hide-subs [hide-subs% 2/100])
|
||||||
|
(unless (member order '(topological self total))
|
||||||
|
(raise-argument-error 'render "(or/c 'topological 'self 'total)" order))
|
||||||
|
(define key (if (eq? order 'total) node-total node-self))
|
||||||
(define (show . xs)
|
(define (show . xs)
|
||||||
(let loop ([x xs])
|
(let loop ([x xs])
|
||||||
(cond [(or (not x) (null? x) (void? x)) (void)]
|
(cond [(or (not x) (null? x) (void? x)) (void)]
|
||||||
|
@ -58,7 +62,10 @@
|
||||||
(define threads+times (profile-thread-times profile))
|
(define threads+times (profile-thread-times profile))
|
||||||
(define *-node (profile-*-node profile))
|
(define *-node (profile-*-node profile))
|
||||||
(define hidden (get-hidden profile hide-self% hide-subs%))
|
(define hidden (get-hidden profile hide-self% hide-subs%))
|
||||||
(define nodes (remq* hidden (profile-nodes profile)))
|
(define nodes (let ([incnodes (remq* hidden (profile-nodes profile))])
|
||||||
|
(if (eq? order 'topological)
|
||||||
|
incnodes
|
||||||
|
(sort incnodes > #:key key))))
|
||||||
(define node->
|
(define node->
|
||||||
(let ([t (make-hasheq)])
|
(let ([t (make-hasheq)])
|
||||||
(for ([node (in-list nodes)] [idx (in-naturals 1)])
|
(for ([node (in-list nodes)] [idx (in-naturals 1)])
|
||||||
|
|
Loading…
Reference in New Issue
Block a user