Skip to content
Snippets Groups Projects
Commit 094d47bf authored by Todd Gamblin's avatar Todd Gamblin Committed by GitHub
Browse files

Allow user to specify profile sort column on the command line. (#4056)

- Add -P <STAT> argument so that caller can specify a sort column for
  cProfile. Can specify multiple columns with commas. e.g.:
      spack -P cumtime,module

- Add --lines option to Spack spec to control number of profile lines
  displayed

- Sort by time by default (because it works in all Python versions)

- Show sort column options in command help.

- Do a short profile run in the unit tests.
parent c86b53a7
No related branches found
No related tags found
No related merge requests found
...@@ -93,6 +93,12 @@ from llnl.util.tty.color import * ...@@ -93,6 +93,12 @@ from llnl.util.tty.color import *
import spack import spack
from spack.error import SpackError from spack.error import SpackError
import argparse import argparse
import pstats
# Get the allowed names of statistics for cProfile, and make a list of
# groups of 7 names to wrap them nicely.
stat_names = pstats.Stats.sort_arg_dict_default
stat_lines = list(zip(*(iter(stat_names),)*7))
# Command parsing # Command parsing
parser = argparse.ArgumentParser( parser = argparse.ArgumentParser(
...@@ -120,10 +126,15 @@ parser.add_argument('-m', '--mock', action='store_true', ...@@ -120,10 +126,15 @@ parser.add_argument('-m', '--mock', action='store_true',
help="use mock packages instead of real ones") help="use mock packages instead of real ones")
parser.add_argument('-p', '--profile', action='store_true', parser.add_argument('-p', '--profile', action='store_true',
help="profile execution using cProfile") help="profile execution using cProfile")
parser.add_argument('-P', '--sorted-profile', default=None, metavar="STAT",
help="profile and sort by one or more of:\n[%s]" %
',\n '.join([', '.join(line) for line in stat_lines]))
parser.add_argument('--lines', default=20, action='store',
help="lines of profile output: default 20; 'all' for all")
parser.add_argument('-v', '--verbose', action='store_true', parser.add_argument('-v', '--verbose', action='store_true',
help="print additional output during builds") help="print additional output during builds")
parser.add_argument('-s', '--stacktrace', action='store_true', parser.add_argument('-s', '--stacktrace', action='store_true',
help="add stacktrace information to all printed statements") help="add stacktrace info to all printed statements")
parser.add_argument('-V', '--version', action='version', parser.add_argument('-V', '--version', action='version',
version="%s" % spack.spack_version) version="%s" % spack.spack_version)
...@@ -206,9 +217,37 @@ def main(args): ...@@ -206,9 +217,37 @@ def main(args):
# actually parse the args. # actually parse the args.
args, unknown = parser.parse_known_args() args, unknown = parser.parse_known_args()
if args.profile: if args.profile or args.sorted_profile:
import cProfile import cProfile
cProfile.runctx('_main(args, unknown)', globals(), locals())
try:
nlines = int(args.lines)
except ValueError:
if args.lines != 'all':
tty.die('Invalid number for --lines: %s' % args.lines)
nlines = -1
# allow comma-separated list of fields
sortby = ['time']
if args.sorted_profile:
sortby = args.sorted_profile.split(',')
for stat in sortby:
if stat not in stat_names:
tty.die("Invalid sort field: %s" % stat)
try:
# make a profiler and run the code.
pr = cProfile.Profile()
pr.enable()
_main(args, unknown)
finally:
pr.disable()
# print out profile stats.
stats = pstats.Stats(pr)
stats.sort_stats(*sortby)
stats.print_stats(nlines)
elif args.pdb: elif args.pdb:
import pdb import pdb
pdb.runctx('_main(args, unknown)', globals(), locals()) pdb.runctx('_main(args, unknown)', globals(), locals())
......
...@@ -20,6 +20,9 @@ cd "$SPACK_ROOT" ...@@ -20,6 +20,9 @@ cd "$SPACK_ROOT"
# Print compiler information # Print compiler information
spack config get compilers spack config get compilers
# Profile and print top 20 lines for a simple call to spack spec
${coverage_run} bin/spack -p --lines 20 spec mpileaks
# Run unit tests with code coverage # Run unit tests with code coverage
${coverage_run} bin/spack test "$@" ${coverage_run} bin/spack test "$@"
${coverage_combine} ${coverage_combine}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment