aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/scripts/flame_graph.sh
blob: ae9129c9242dda17b5ebf3ca432ce9b68a38176a (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
#!/usr/bin/env bash

# Copyright (c) 2023 Egor Tensin <Egor.Tensin@gmail.com>
# This file is part of the "cimple" project.
# For details, see https://github.com/egor-tensin/cimple.
# Distributed under the MIT License.

# This script attaches to a set of processes and makes a flame graph using
# flamegraph.pl.

set -o errexit -o nounset -o pipefail
shopt -s inherit_errexit lastpipe

script_name="$( basename -- "${BASH_SOURCE[0]}" )"
readonly script_name

script_usage() {
    local msg
    for msg; do
        echo "$script_name: $msg"
    done

    echo "usage: $script_name OUTPUT_PATH PID [PID...]"
}

join_pids() {
    local result=''
    while [ "$#" -gt 0 ]; do
        if [ -n "$result" ]; then
            result="$result,"
        fi
        result="$result$1"
        shift
    done
    echo "$result"
}

output_dir=''

cleanup() {
    if [ -n "$output_dir" ]; then
        echo "Removing temporary directory: $output_dir"
        rm -rf -- "$output_dir"
    fi
}

make_graph() {
    wait "$record_pid" || true
    perf script -i "$output_dir/perf.data" > "$output_dir/perf.out"
    stackcollapse-perf.pl "$output_dir/perf.out" > "$output_dir/perf.folded"
    flamegraph.pl --width 2400 "$output_dir/perf.folded" > "$output_path"
}

record_pid=''

stop_record() {
    echo "Stopping 'perf record' process $record_pid"
    kill -SIGINT "$record_pid"
    make_graph
}

check_tools() {
    local tool
    for tool in perf stackcollapse-perf.pl flamegraph.pl; do
        if ! command -v "$tool" &> /dev/null; then
            echo "$script_name: $tool is missing" >&2
            exit 1
        fi
    done
}

main() {
    trap cleanup EXIT
    check_tools

    if [ "$#" -lt 1 ]; then
        script_usage "output path is required" >&2
        exit 1
    fi
    local output_path="$1"
    output_path="$( realpath -- "$output_path" )"
    shift

    if [ "$#" -lt 1 ]; then
        script_usage "at least one process ID is required" >&2
        exit 1
    fi
    local pids
    pids="$( join_pids "$@" )"
    shift

    echo "Output path: $output_path"
    echo "PIDs: $pids"

    output_dir="$( dirname -- "$output_path" )"
    output_dir="$( mktemp -d --tmpdir="$output_dir" )"
    readonly output_dir

    perf record \
        -o "$output_dir/perf.data" \
        --freq=99 \
        --call-graph dwarf,65528 \
        --pid="$pids" \
        --no-inherit &

    record_pid="$!"
    echo "Started 'perf record' process $record_pid"
    trap stop_record SIGINT SIGTERM

    make_graph
}

main "$@"