2014-07-31 10:01:12 +04:00
#!/bin/bash
# perf-with-kcore: use perf with a copy of kcore
# Copyright (c) 2014, Intel Corporation.
#
# This program is free software; you can redistribute it and/or modify it
# under the terms and conditions of the GNU General Public License,
# version 2, as published by the Free Software Foundation.
#
# This program is distributed in the hope it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
# more details.
set -e
usage( )
{
echo "Usage: perf-with-kcore <perf sub-command> <perf.data directory> [<sub-command options> [ -- <workload>]]" >& 2
echo " <perf sub-command> can be record, script, report or inject" >& 2
echo " or: perf-with-kcore fix_buildid_cache_permissions" >& 2
exit 1
}
find_perf( )
{
if [ -n " $PERF " ] ; then
return
fi
PERF = ` which perf || true `
if [ -z " $PERF " ] ; then
echo "Failed to find perf" >& 2
exit 1
fi
if [ ! -x " $PERF " ] ; then
echo "Failed to find perf" >& 2
exit 1
fi
echo " Using $PERF "
" $PERF " version
}
copy_kcore( )
{
echo "Copying kcore"
if [ $EUID -eq 0 ] ; then
SUDO = ""
else
SUDO = "sudo"
fi
rm -f perf.data.junk
2015-07-17 19:33:47 +03:00
( " $PERF " record -o perf.data.junk " ${ PERF_OPTIONS [@] } " -- sleep 60) >/dev/null 2>/dev/null &
2014-07-31 10:01:12 +04:00
PERF_PID = $!
# Need to make sure that perf has started
sleep 1
KCORE = $(( $SUDO " $PERF " buildid-cache - v - f - k / proc/kcore >/dev/null) 2 >& 1 )
case " $KCORE " in
"kcore added to build-id cache directory " *)
KCORE_DIR = ${ KCORE # "kcore added to build-id cache directory " }
; ;
*)
kill $PERF_PID
wait >/dev/null 2>/dev/null || true
rm perf.data.junk
echo " $KCORE "
echo "Failed to find kcore" >& 2
exit 1
; ;
esac
kill $PERF_PID
wait >/dev/null 2>/dev/null || true
rm perf.data.junk
$SUDO cp -a " $KCORE_DIR " " $( pwd ) / $PERF_DATA_DIR "
$SUDO rm -f " $KCORE_DIR /kcore "
$SUDO rm -f " $KCORE_DIR /kallsyms "
$SUDO rm -f " $KCORE_DIR /modules "
$SUDO rmdir " $KCORE_DIR "
KCORE_DIR_BASENAME = $( basename " $KCORE_DIR " )
KCORE_DIR = " $( pwd ) / $PERF_DATA_DIR / $KCORE_DIR_BASENAME "
$SUDO chown $UID " $KCORE_DIR "
$SUDO chown $UID " $KCORE_DIR /kcore "
$SUDO chown $UID " $KCORE_DIR /kallsyms "
$SUDO chown $UID " $KCORE_DIR /modules "
$SUDO chgrp $GROUPS " $KCORE_DIR "
$SUDO chgrp $GROUPS " $KCORE_DIR /kcore "
$SUDO chgrp $GROUPS " $KCORE_DIR /kallsyms "
$SUDO chgrp $GROUPS " $KCORE_DIR /modules "
ln -s " $KCORE_DIR_BASENAME " " $PERF_DATA_DIR /kcore_dir "
}
fix_buildid_cache_permissions( )
{
if [ $EUID -ne 0 ] ; then
echo "This script must be run as root via sudo " >& 2
exit 1
fi
if [ -z " $SUDO_USER " ] ; then
echo "This script must be run via sudo" >& 2
exit 1
fi
USER_HOME = $( bash <<< " echo ~ $SUDO_USER " )
if [ " $HOME " != " $USER_HOME " ] ; then
echo " Fix unnecessary because root has a home: $HOME " >& 2
exit 1
fi
echo "Fixing buildid cache permissions"
find " $USER_HOME /.debug " -xdev -type d ! -user " $SUDO_USER " -ls -exec chown " $SUDO_USER " \{ \} \;
find " $USER_HOME /.debug " -xdev -type f -links 1 ! -user " $SUDO_USER " -ls -exec chown " $SUDO_USER " \{ \} \;
find " $USER_HOME /.debug " -xdev -type l ! -user " $SUDO_USER " -ls -exec chown -h " $SUDO_USER " \{ \} \;
if [ -n " $SUDO_GID " ] ; then
find " $USER_HOME /.debug " -xdev -type d ! -group " $SUDO_GID " -ls -exec chgrp " $SUDO_GID " \{ \} \;
find " $USER_HOME /.debug " -xdev -type f -links 1 ! -group " $SUDO_GID " -ls -exec chgrp " $SUDO_GID " \{ \} \;
find " $USER_HOME /.debug " -xdev -type l ! -group " $SUDO_GID " -ls -exec chgrp -h " $SUDO_GID " \{ \} \;
fi
echo "Done"
}
check_buildid_cache_permissions( )
{
if [ $EUID -eq 0 ] ; then
return
fi
PERMISSIONS_OK += $( find " $HOME /.debug " -xdev -type d ! -user " $USER " -print -quit)
PERMISSIONS_OK += $( find " $HOME /.debug " -xdev -type f -links 1 ! -user " $USER " -print -quit)
PERMISSIONS_OK += $( find " $HOME /.debug " -xdev -type l ! -user " $USER " -print -quit)
PERMISSIONS_OK += $( find " $HOME /.debug " -xdev -type d ! -group " $GROUPS " -print -quit)
PERMISSIONS_OK += $( find " $HOME /.debug " -xdev -type f -links 1 ! -group " $GROUPS " -print -quit)
PERMISSIONS_OK += $( find " $HOME /.debug " -xdev -type l ! -group " $GROUPS " -print -quit)
if [ -n " $PERMISSIONS_OK " ] ; then
echo "*** WARNING *** buildid cache permissions may need fixing" >& 2
fi
}
record( )
{
echo "Recording"
if [ $EUID -ne 0 ] ; then
if [ " $( cat /proc/sys/kernel/kptr_restrict) " -ne 0 ] ; then
echo "*** WARNING *** /proc/sys/kernel/kptr_restrict prevents access to kernel addresses" >& 2
fi
2015-07-17 19:33:47 +03:00
if echo " ${ PERF_OPTIONS [@] } " | grep -q ' -a \|^-a \| -a$\|^-a$\| --all-cpus \|^--all-cpus \| --all-cpus$\|^--all-cpus$' ; then
2014-07-31 10:01:12 +04:00
echo "*** WARNING *** system-wide tracing without root access will not be able to read all necessary information from /proc" >& 2
fi
2015-07-17 19:33:47 +03:00
if echo " ${ PERF_OPTIONS [@] } " | grep -q 'intel_pt\|intel_bts\| -I\|^-I' ; then
2014-07-31 10:01:12 +04:00
if [ " $( cat /proc/sys/kernel/perf_event_paranoid) " -gt -1 ] ; then
echo "*** WARNING *** /proc/sys/kernel/perf_event_paranoid restricts buffer size and tracepoint (sched_switch) use" >& 2
fi
2015-07-17 19:33:47 +03:00
if echo " ${ PERF_OPTIONS [@] } " | grep -q ' --per-thread \|^--per-thread \| --per-thread$\|^--per-thread$' ; then
2014-07-31 10:01:12 +04:00
true
2015-07-17 19:33:47 +03:00
elif echo " ${ PERF_OPTIONS [@] } " | grep -q ' -t \|^-t \| -t$\|^-t$' ; then
2014-07-31 10:01:12 +04:00
true
elif [ ! -r /sys/kernel/debug -o ! -x /sys/kernel/debug ] ; then
echo "*** WARNING *** /sys/kernel/debug permissions prevent tracepoint (sched_switch) use" >& 2
fi
fi
fi
if [ -z " $1 " ] ; then
echo "Workload is required for recording" >& 2
usage
fi
if [ -e " $PERF_DATA_DIR " ] ; then
echo " ' $PERF_DATA_DIR ' exists " >& 2
exit 1
fi
find_perf
mkdir " $PERF_DATA_DIR "
2015-07-17 19:33:47 +03:00
echo " $PERF record -o $PERF_DATA_DIR /perf.data ${ PERF_OPTIONS [@] } -- $@ "
" $PERF " record -o " $PERF_DATA_DIR /perf.data " " ${ PERF_OPTIONS [@] } " -- " $@ " || true
2014-07-31 10:01:12 +04:00
if rmdir " $PERF_DATA_DIR " > /dev/null 2>/dev/null ; then
exit 1
fi
copy_kcore
echo "Done"
}
subcommand( )
{
find_perf
check_buildid_cache_permissions
2015-07-17 19:33:47 +03:00
echo " $PERF $PERF_SUB_COMMAND -i $PERF_DATA_DIR /perf.data --kallsyms= $PERF_DATA_DIR /kcore_dir/kallsyms $@ "
" $PERF " $PERF_SUB_COMMAND -i " $PERF_DATA_DIR /perf.data " " --kallsyms= $PERF_DATA_DIR /kcore_dir/kallsyms " " $@ "
2014-07-31 10:01:12 +04:00
}
if [ " $1 " = "fix_buildid_cache_permissions" ] ; then
fix_buildid_cache_permissions
exit 0
fi
PERF_SUB_COMMAND = $1
PERF_DATA_DIR = $2
shift || true
shift || true
if [ -z " $PERF_SUB_COMMAND " ] ; then
usage
fi
if [ -z " $PERF_DATA_DIR " ] ; then
usage
fi
case " $PERF_SUB_COMMAND " in
"record" )
while [ " $1 " != "--" ] ; do
2015-07-17 19:33:47 +03:00
PERF_OPTIONS += ( " $1 " )
2014-07-31 10:01:12 +04:00
shift || break
done
if [ " $1 " != "--" ] ; then
echo "Options and workload are required for recording" >& 2
usage
fi
shift
2015-07-17 19:33:47 +03:00
record " $@ "
2014-07-31 10:01:12 +04:00
; ;
"script" )
2015-07-17 19:33:47 +03:00
subcommand " $@ "
2014-07-31 10:01:12 +04:00
; ;
"report" )
2015-07-17 19:33:47 +03:00
subcommand " $@ "
2014-07-31 10:01:12 +04:00
; ;
"inject" )
2015-07-17 19:33:47 +03:00
subcommand " $@ "
2014-07-31 10:01:12 +04:00
; ;
*)
usage
; ;
esac