Add options to control the order of profile output.

This commit is contained in:
James Bornholt 2016-07-02 20:18:44 -07:00
parent 81bb14ebf1
commit 109ee25287
6 changed files with 60 additions and 10 deletions

View File

@ -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.}

View File

@ -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 ...)]{

View File

@ -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 ()

View File

@ -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?)]))

View File

@ -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))

View File

@ -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)])