#!/usr/bin/env perl # Copyright 2008, Intel Corporation # # This file is part of the Linux kernel # # This program file is free software; you can redistribute it and/or modify it # under the terms of the GNU General Public License as published by the # Free Software Foundation; version 2 of the License. # # This program is distributed in the hope that 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. # # You should have received a copy of the GNU General Public License # along with this program in a file named COPYING; if not, write to the # Free Software Foundation, Inc., # 51 Franklin Street, Fifth Floor, # Boston, MA 02110-1301 USA # # Authors: # Arjan van de Ven <arjan@linux.intel.com> # # This script turns a dmesg output into a SVG graphic that shows which # functions take how much time. You can view SVG graphics with various # programs, including Inkscape, The Gimp and Firefox. # # # For this script to work, the kernel needs to be compiled with the # CONFIG_PRINTK_TIME configuration option enabled, and with # "initcall_debug" passed on the kernel command line. # # usage: # dmesg | perl scripts/bootgraph.pl > output.svg # use strict; use Getopt::Long; my $header = 0; sub help { my $text = << "EOM"; Usage: 1) dmesg | perl scripts/bootgraph.pl [OPTION] > output.svg 2) perl scripts/bootgraph.pl -h Options: -header Insert kernel version and date EOM my $std=shift; if ($std == 1) { print STDERR $text; } else { print $text; } exit; } GetOptions( 'h|help' =>\&help, 'header' =>\$header ); my %start; my %end; my %type; my $done = 0; my $maxtime = 0; my $firsttime = 99999; my $count = 0; my %pids; my %pidctr; my $headerstep = 20; my $xheader = 15; my $yheader = 25; my $cyheader = 0; while (<>) { my $line = $_; if ($line =~ /([0-9\.]+)\] calling ([a-zA-Z0-9\_\.]+)\+/) { my $func = $2; if ($done == 0) { $start{$func} = $1; $type{$func} = 0; if ($1 < $firsttime) { $firsttime = $1; } } if ($line =~ /\@ ([0-9]+)/) { $pids{$func} = $1; } $count = $count + 1; } if ($line =~ /([0-9\.]+)\] async_waiting @ ([0-9]+)/) { my $pid = $2; my $func; if (!defined($pidctr{$pid})) { $func = "wait_" . $pid . "_1"; $pidctr{$pid} = 1; } else { $pidctr{$pid} = $pidctr{$pid} + 1; $func = "wait_" . $pid . "_" . $pidctr{$pid}; } if ($done == 0) { $start{$func} = $1; $type{$func} = 1; if ($1 < $firsttime) { $firsttime = $1; } } $pids{$func} = $pid; $count = $count + 1; } if ($line =~ /([0-9\.]+)\] initcall ([a-zA-Z0-9\_\.]+)\+.*returned/) { if ($done == 0) { $end{$2} = $1; $maxtime = $1; } } if ($line =~ /([0-9\.]+)\] async_continuing @ ([0-9]+)/) { my $pid = $2; my $func = "wait_" . $pid . "_" . $pidctr{$pid}; $end{$func} = $1; $maxtime = $1; } if ($line =~ /Write protecting the/) { $done = 1; } if ($line =~ /Freeing unused kernel memory/) { $done = 1; } } if ($count == 0) { print STDERR <<END; No data found in the dmesg. Make sure that 'printk.time=1' and 'initcall_debug' are passed on the kernel command line. END help(1); exit 1; } print "<?xml version=\"1.0\" standalone=\"no\"?> \n"; print "<svg width=\"2000\" height=\"100%\" version=\"1.1\" xmlns=\"http://www.w3.org/2000/svg\">\n"; if ($header) { my $version = `uname -a`; my $date = `date`; print "<text transform=\"translate($xheader,$yheader)\">Kernel version: $version</text>\n"; $cyheader = $yheader+$headerstep; print "<text transform=\"translate($xheader,$cyheader)\">Date: $date</text>\n"; } my @styles; $styles[0] = "fill:rgb(0,0,255);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[1] = "fill:rgb(0,255,0);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[2] = "fill:rgb(255,0,20);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[3] = "fill:rgb(255,255,20);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[4] = "fill:rgb(255,0,255);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[5] = "fill:rgb(0,255,255);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[6] = "fill:rgb(0,128,255);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[7] = "fill:rgb(0,255,128);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[8] = "fill:rgb(255,0,128);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[9] = "fill:rgb(255,255,128);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[10] = "fill:rgb(255,128,255);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; $styles[11] = "fill:rgb(128,255,255);fill-opacity:0.5;stroke-width:1;stroke:rgb(0,0,0)"; my $style_wait = "fill:rgb(128,128,128);fill-opacity:0.5;stroke-width:0;stroke:rgb(0,0,0)"; my $mult = 1950.0 / ($maxtime - $firsttime); my $threshold2 = ($maxtime - $firsttime) / 120.0; my $threshold = $threshold2/10; my $stylecounter = 0; my %rows; my $rowscount = 1; my @initcalls = sort { $start{$a} <=> $start{$b} } keys(%start); foreach my $key (@initcalls) { my $duration = $end{$key} - $start{$key}; if ($duration >= $threshold) { my ($s, $s2, $s3, $e, $w, $y, $y2, $style); my $pid = $pids{$key}; if (!defined($rows{$pid})) { $rows{$pid} = $rowscount; $rowscount = $rowscount + 1; } $s = ($start{$key} - $firsttime) * $mult; $s2 = $s + 6; $s3 = $s + 1; $e = ($end{$key} - $firsttime) * $mult; $w = $e - $s; $y = $rows{$pid} * 150; $y2 = $y + 4; $style = $styles[$stylecounter]; $stylecounter = $stylecounter + 1; if ($stylecounter > 11) { $stylecounter = 0; }; if ($type{$key} == 1) { $y = $y + 15; print "<rect x=\"$s\" width=\"$w\" y=\"$y\" height=\"115\" style=\"$style_wait\"/>\n"; } else { print "<rect x=\"$s\" width=\"$w\" y=\"$y\" height=\"145\" style=\"$style\"/>\n"; if ($duration >= $threshold2) { print "<text transform=\"translate($s2,$y2) rotate(90)\">$key</text>\n"; } else { print "<text transform=\"translate($s3,$y2) rotate(90)\" font-size=\"3pt\">$key</text>\n"; } } } } # print the time line on top my $time = $firsttime; my $step = ($maxtime - $firsttime) / 15; while ($time < $maxtime) { my $s3 = ($time - $firsttime) * $mult; my $tm = int($time * 100) / 100.0; print "<text transform=\"translate($s3,89) rotate(90)\">$tm</text>\n"; $time = $time + $step; } print "</svg>\n";
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
basic | Folder | 0755 |
|
|
coccinelle | Folder | 0755 |
|
|
dtc | Folder | 0755 |
|
|
gcc-plugins | Folder | 0755 |
|
|
gdb | Folder | 0755 |
|
|
genksyms | Folder | 0755 |
|
|
kconfig | Folder | 0755 |
|
|
ksymoops | Folder | 0755 |
|
|
mod | Folder | 0755 |
|
|
package | Folder | 0755 |
|
|
selinux | Folder | 0755 |
|
|
tracing | Folder | 0755 |
|
|
.gitignore | File | 162 B | 0644 |
|
Kbuild.include | File | 18.36 KB | 0644 |
|
Lindent | File | 502 B | 0755 |
|
Makefile | File | 1.9 KB | 0644 |
|
Makefile.asm-generic | File | 1.17 KB | 0644 |
|
Makefile.build | File | 20.68 KB | 0644 |
|
Makefile.clean | File | 2.96 KB | 0644 |
|
Makefile.dtbinst | File | 1.09 KB | 0644 |
|
Makefile.extrawarn | File | 2.9 KB | 0644 |
|
Makefile.gcc-plugins | File | 3.75 KB | 0644 |
|
Makefile.headersinst | File | 3.98 KB | 0644 |
|
Makefile.host | File | 6.23 KB | 0644 |
|
Makefile.kasan | File | 1.09 KB | 0644 |
|
Makefile.kcov | File | 201 B | 0644 |
|
Makefile.lib | File | 15.25 KB | 0644 |
|
Makefile.modbuiltin | File | 1.82 KB | 0644 |
|
Makefile.modinst | File | 1.52 KB | 0644 |
|
Makefile.modpost | File | 5.43 KB | 0644 |
|
Makefile.modsign | File | 1.02 KB | 0644 |
|
Makefile.ubsan | File | 1.07 KB | 0644 |
|
adjust_autoksyms.sh | File | 3.09 KB | 0755 |
|
asn1_compiler.c | File | 35.54 KB | 0644 |
|
bloat-o-meter | File | 3.22 KB | 0755 |
|
bootgraph.pl | File | 6.28 KB | 0755 |
|
check_00index.sh | File | 1.3 KB | 0755 |
|
check_extable.sh | File | 4.93 KB | 0755 |
|
checkincludes.pl | File | 1.94 KB | 0755 |
|
checkkconfigsymbols.py | File | 15.51 KB | 0755 |
|
checkpatch.pl | File | 186.79 KB | 0755 |
|
checkstack.pl | File | 5.49 KB | 0755 |
|
checksyscalls.sh | File | 5.68 KB | 0755 |
|
checkversion.pl | File | 1.9 KB | 0755 |
|
cleanfile | File | 3.46 KB | 0755 |
|
cleanpatch | File | 5.06 KB | 0755 |
|
coccicheck | File | 7.24 KB | 0755 |
|
config | File | 4.64 KB | 0755 |
|
conmakehash.c | File | 5.98 KB | 0644 |
|
const_structs.checkpatch | File | 964 B | 0644 |
|
decode_stacktrace.sh | File | 3.82 KB | 0755 |
|
decodecode | File | 2.3 KB | 0755 |
|
depmod.sh | File | 1.94 KB | 0755 |
|
diffconfig | File | 3.72 KB | 0755 |
|
documentation-file-ref-check | File | 395 B | 0755 |
|
export_report.pl | File | 4.55 KB | 0755 |
|
extract-cert.c | File | 3.63 KB | 0644 |
|
extract-ikconfig | File | 1.69 KB | 0755 |
|
extract-module-sig.pl | File | 3.66 KB | 0755 |
|
extract-sys-certs.pl | File | 3.75 KB | 0755 |
|
extract-vmlinux | File | 1.6 KB | 0755 |
|
extract_xc3028.pl | File | 44.6 KB | 0755 |
|
faddr2line | File | 5.53 KB | 0755 |
|
find-unused-docs.sh | File | 1.27 KB | 0755 |
|
gcc-goto.sh | File | 530 B | 0755 |
|
gcc-ld | File | 711 B | 0755 |
|
gcc-plugin.sh | File | 1.06 KB | 0755 |
|
gcc-version.sh | File | 857 B | 0755 |
|
gcc-x86_32-has-stack-protector.sh | File | 219 B | 0755 |
|
gcc-x86_64-has-stack-protector.sh | File | 244 B | 0755 |
|
gen_initramfs_list.sh | File | 8.03 KB | 0755 |
|
get_dvb_firmware | File | 25.22 KB | 0755 |
|
get_maintainer.pl | File | 65.03 KB | 0755 |
|
gfp-translate | File | 1.71 KB | 0755 |
|
headerdep.pl | File | 3.5 KB | 0755 |
|
headers.sh | File | 512 B | 0755 |
|
headers_check.pl | File | 3.73 KB | 0755 |
|
headers_install.sh | File | 1.32 KB | 0755 |
|
insert-sys-cert.c | File | 13.08 KB | 0644 |
|
kallsyms.c | File | 18.89 KB | 0644 |
|
kernel-doc | File | 92.36 KB | 0755 |
|
kmsg-doc | File | 12.13 KB | 0755 |
|
ld-version.sh | File | 269 B | 0755 |
|
leaking_addresses.pl | File | 9.65 KB | 0755 |
|
link-vmlinux.sh | File | 7.65 KB | 0755 |
|
makelst | File | 808 B | 0755 |
|
markup_oops.pl | File | 8.08 KB | 0755 |
|
mkcompile_h | File | 2.74 KB | 0755 |
|
mkmakefile | File | 1.19 KB | 0755 |
|
mksysmap | File | 1.34 KB | 0755 |
|
mkuboot.sh | File | 414 B | 0755 |
|
module-common.lds | File | 901 B | 0644 |
|
namespace.pl | File | 13.18 KB | 0755 |
|
objdiff | File | 2.85 KB | 0755 |
|
parse-maintainers.pl | File | 3.72 KB | 0755 |
|
patch-kernel | File | 9.95 KB | 0755 |
|
pnmtologo.c | File | 11.91 KB | 0644 |
|
profile2linkerlist.pl | File | 414 B | 0755 |
|
prune-kernel | File | 708 B | 0755 |
|
recordmcount.c | File | 17.49 KB | 0644 |
|
recordmcount.h | File | 16.4 KB | 0644 |
|
recordmcount.pl | File | 18.41 KB | 0755 |
|
setlocalversion | File | 4.72 KB | 0755 |
|
show_delta | File | 2.99 KB | 0755 |
|
sign-file.c | File | 9.93 KB | 0644 |
|
sortextable.c | File | 8.35 KB | 0644 |
|
sortextable.h | File | 5.48 KB | 0644 |
|
spelling.txt | File | 24.97 KB | 0644 |
|
sphinx-pre-install | File | 14.04 KB | 0755 |
|
stackdelta | File | 1.84 KB | 0755 |
|
stackusage | File | 794 B | 0755 |
|
subarch.include | File | 641 B | 0644 |
|
tags.sh | File | 9.49 KB | 0755 |
|
ubuntu-retpoline-extract-one | File | 7.26 KB | 0644 |
|
unifdef.c | File | 34.8 KB | 0644 |
|
ver_linux | File | 2.94 KB | 0755 |
|
xen-hypercalls.sh | File | 386 B | 0644 |
|
xz_wrap.sh | File | 562 B | 0755 |
|