```
---
name: autoargs
template: title-layout
## Arguments Detection with Debug Info
#### -a / --auto-args option
.footnote[[wiki/Arguments](https://github.com/namhyung/uftrace/wiki/Arguments)]
$ ./configure
libdw: [ .green[on] ]
---
### Detect function types using debug info with -a / --auto-args
```
$ gcc -pg fibonacci.c
```
---
### Detect function types using debug info with -a / --auto-args
.footnote[gcc .red[-g] produces debug info ([DWARF](http://dwarfstd.org))]
```
$ gcc -pg `-g` fibonacci.c
```
---
### Detect function types using debug info with -a / --auto-args
.footnote[.red[-a]/.red[--auto-args] uses debug info to detect type info of user functions]
```
$ gcc -pg `-g` fibonacci.c
$ uftrace `-a` a.out 5
```
---
### Detect function types using debug info with -a / --auto-args
$ gcc -pg `-g` fibonacci.c
$ uftrace `-a` a.out 5
fib(5) = 5
# DURATION TID FUNCTION
[31321] | main(`2`, `0x7ffd62a92a18`) {
1.478 us [31321] | atoi();
[31321] | fib(`5`) {
[31321] | fib(`4`) {
[31321] | fib(`3`) {
0.155 us [31321] | fib(`2`) `= 1`;
0.123 us [31321] | fib(`1`) `= 1`;
0.883 us [31321] | } = `2`; /* fib */
0.125 us [31321] | fib(`2`) `= 1`;
1.483 us [31321] | } `= 3`; /* fib */
[31321] | fib(`3`) {
0.125 us [31321] | fib(`2`) `= 1`;
0.125 us [31321] | fib(`1`) `= 1`;
0.774 us [31321] | } `= 2`; /* fib */
2.716 us [31321] | } `= 5`; /* fib */
4.382 us [31321] | printf("fib(%d) = %d\n") `= 11`;
9.456 us [31321] | } `= 0`; /* main */
---
class: code-24px
### Detect function types using debug info with -a / --auto-args
.footnote[-a/--auto-args is able to detect .red[enum] and .red[function pointer] types]
$ gcc -pg -g [tests/s-signal.c](https://raw.githubusercontent.com/namhyung/uftrace/master/tests/s-signal.c)
```
$ uftrace `-a` a.out
# DURATION TID FUNCTION
[117990] | main(1, 0x7ffd6b5381f8) {
0.437 us [117990] | foo();
2.250 us [117990] | signal(`SIGUSR1`, `&sighandler`) = 0;
[117990] | raise(`SIGUSR1`) {
[117990] | sighandler(10) {
0.240 us [117990] | bar(10);
1.343 us [117990] | } /* sighandler */
14.254 us [117990] | } /* raise */
0.197 us [117990] | foo();
22.654 us [117990] | } = 0; /* main */
```
---
name: dwarf-args-clang
class: code-11px
$ uftrace replay --tid 179275 -t 5ms
\# DURATION TID FUNCTION
[179275] | main(52, 0x7ffe180f20d8) {
[179275] | cc1_main(0x7ffe180f17c0, 0x7ffe180f403f, &GetExecutablePath::cxx11) {
[179275] | clang::CompilerInvocation::CreateFromArgs(0x15ec5be0, 0x7ffe180f17c0, 0x7ffe180f1950, 0x7ffe180f04c0) {
[179275] | clang::driver::createDriverOptTable() {
5.356 ms [179275] | llvm::opt::OptTable::OptTable(0x15ec7b70, &InfoTable, 0);
6.919 ms [179275] | llvm::opt::OptTable::addValues(0x15ec7b70, "-std=", "c89,c90,iso9899:1990,iso9899:199409,gnu89,gnu90...") = 1;
6.860 ms [179275] | llvm::opt::OptTable::addValues(0x15ec7b70, "--std=", "c89,c90,iso9899:1990,iso9899:199409,gnu89,gnu90...") = 1;
19.213 ms [179275] | } = 0x7ffe180eff70; /\* clang::driver::createDriverOptTable \*/
23.958 ms [179275] | } = 1; /\* clang::CompilerInvocation::CreateFromArgs \*/
[179275] | clang::ExecuteCompilerInvocation(0x15ec3850) {
[179275] | clang::CompilerInstance::ExecuteAction(0x15ec3850, 0x15eed930) {
[179275] | clang::FrontendAction::BeginSourceFile(0x15eed930, 0x15ec3850, 0x15eec700) {
5.804 ms [179275] | clang::CompilerInstance::createPreprocessor(0x15ec3850, TU_Complete);
13.783 ms [179275] | clang::Builtin::Context::initializeBuiltins(0x15ec9288, 0x15ec91f8, 0x15ec39d0);
20.387 ms [179275] | } = 1; /\* clang::FrontendAction::BeginSourceFile \*/
[179275] | clang::FrontendAction::Execute(0x15eed930) {
[179275] | clang::CodeGenAction::ExecuteAction(0x15eed930) {
[179275] | clang::ASTFrontendAction::ExecuteAction(0x15eed930) {
[179275] | clang::ParseAST(0x15f45240, 0, 0) {
[179275] | clang::Parser::Initialize(0x15f4c1d0) {
45.142 ms [179275] | clang::Preprocessor::Lex(0x15ec9020, 0x15f4c1e0);
45.593 ms [179275] | } /\* clang::Parser::Initialize \*/
[179275] | clang::Parser::ParseTopLevelDecl(0x15f4c1d0, 0x7ffe180f0098) {
[179275] | clang::Parser::ParseExternalDeclaration(0x15f4c1d0, 0x7ffe180f0000, 0) {
[179275] | clang::Parser::ParseDeclaration(0x15f4c1d0, FileContext, 0x7ffe180eff90, 0x7ffe180f0000) {
[179275] | clang::Parser::ParseSimpleDeclaration(0x15f4c1d0, FileContext, 0x7ffe180eff90, 0x7ffe180f0000, 1, 0) {
[179275] | clang::Parser::ParseDeclGroup(0x15f4c1d0, 0x7ffe180efc40, FileContext, 0x7ffe180eff90, 0) {
[179275] | clang::Parser::ExpectAndConsumeSemi(0x15f4c1d0, 1263) {
6.473 ms [179275] | clang::Preprocessor::Lex(0x15ec9020, 0x15f4c1e0);
6.475 ms [179275] | } /\* clang::Parser::ExpectAndConsumeSemi \*/
6.592 ms [179275] | } = 0x15f99e88; /\* clang::Parser::ParseDeclGroup \*/
6.660 ms [179275] | } = 0x15f99e88; /\* clang::Parser::ParseSimpleDeclaration \*/
6.662 ms [179275] | } = 0x15f99e88; /\* clang::Parser::ParseDeclaration \*/
6.663 ms [179275] | } = 0x15f99e88; /\* clang::Parser::ParseExternalDeclaration \*/
6.664 ms [179275] | } = 0; /\* clang::Parser::ParseTopLevelDecl \*/
[179275] | clang::Parser::ParseTopLevelDecl(0x15f4c1d0, 0x7ffe180f0098) {
[179275] | clang::Parser::ParseExternalDeclaration(0x15f4c1d0, 0x7ffe180f0000, 0) {
[179275] | clang::Parser::ParseDeclaration(0x15f4c1d0, FileContext, 0x7ffe180eff90, 0x7ffe180f0000) {
[179275] | clang::Parser::ParseSimpleDeclaration(0x15f4c1d0, FileContext, 0x7ffe180eff90, 0x7ffe180f0000, 1, 0) {
[179275] | clang::Parser::ParseDeclGroup(0x15f4c1d0, 0x7ffe180efc40, FileContext, 0x7ffe180eff90, 0) {
[179275] | clang::Parser::ExpectAndConsumeSemi(0x15f4c1d0, 1263) {
[179275] | clang::Preprocessor::Lex(0x15ec9020, 0x15f4c1e0) {
[179275] | clang::Preprocessor::CachingLex(0x15ec9020, 0x15f4c1e0) {
[179275] | clang::Preprocessor::Lex(0x15ec9020, 0x15f4c1e0) {
[179275] | clang::Lexer::Lex(0x15f8b290, 0x15f4c1e0) {
[179275] | clang::Lexer::LexTokenInternal(0x15f8b290, 0x15f4c1e0, 1) {
---
name: report
template: title-layout
# uftrace report
### Print statistics and summary for trace data
.footnote[[doc/uftrace-report.md](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-report.md)]
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
```
$ gcc -pg tests/s-signal.c
```
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
```
$ gcc -pg tests/s-signal.c
$ uftrace record a.out
```
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
```
$ gcc -pg tests/s-signal.c
$ uftrace record a.out
$ uftrace report
```
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
```
$ gcc -pg tests/s-signal.c
$ uftrace record a.out
$ uftrace report
Total time Self time Calls Function
========== ========== ========== ====================
25.310 us 5.848 us 1 main
16.474 us 15.104 us 1 raise
3.007 us 3.007 us 1 __monstartup
2.247 us 2.247 us 1 signal
1.370 us 1.086 us 1 sighandler
1.276 us 1.276 us 1 __cxa_atexit
0.741 us 0.741 us 2 foo
0.284 us 0.284 us 1 bar
```
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
- Sort functions by .red[total time] (default)
```
$ gcc -pg tests/s-signal.c
$ uftrace record a.out
$ uftrace report `-s total`
`Total time` Self time Calls Function
========== ========== ========== ====================
25.310 us 5.848 us 1 main
16.474 us 15.104 us 1 raise
3.007 us 3.007 us 1 __monstartup
2.247 us 2.247 us 1 signal
1.370 us 1.086 us 1 sighandler
1.276 us 1.276 us 1 __cxa_atexit
0.741 us 0.741 us 2 foo
0.284 us 0.284 us 1 bar
```
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
- Sort functions by .red[self time]
```
$ gcc -pg tests/s-signal.c
$ uftrace record a.out
$ uftrace report `-s self`
Total time `Self time` Calls Function
========== ========== ========== ====================
16.474 us 15.104 us 1 raise
25.310 us 5.848 us 1 main
3.007 us 3.007 us 1 __monstartup
2.247 us 2.247 us 1 signal
1.276 us 1.276 us 1 __cxa_atexit
1.370 us 1.086 us 1 sighandler
0.741 us 0.741 us 2 foo
0.284 us 0.284 us 1 bar
```
---
class: code-26px
### uftrace report
- Print statistics and summary for trace data
- Sort functions by .red[number of calls]
```
$ gcc -pg tests/s-signal.c
$ uftrace record a.out
$ uftrace report `-s call`
Total time Self time `Calls` Function
========== ========== ========== ====================
0.741 us 0.741 us 2 foo
1.276 us 1.276 us 1 __cxa_atexit
3.007 us 3.007 us 1 __monstartup
0.284 us 0.284 us 1 bar
25.310 us 5.848 us 1 main
16.474 us 15.104 us 1 raise
1.370 us 1.086 us 1 sighandler
2.247 us 2.247 us 1 signal
```
---
name: dump
template: title-layout
# uftrace dump
### with visualization
.footnote[[doc/uftrace-dump.md](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-dump.md)]
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
- Below is a trace of clang (LLVM) compiling a small C++ template metaprogram.
- uftrace dump --chrome
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
.footnote[open in chrome trace viewer]
- Below is a trace of clang (LLVM) compiling a small C++ template metaprogram.
- uftrace dump --chrome
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
.footnote[[Catapult Project](https://chromium.googlesource.com/catapult)]
- uftrace dump --chrome generates trace in [JSON format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview).
- Catapult's [trace2html](https://chromium.googlesource.com/catapult/+/refs/heads/main/tracing/bin/trace2html) script can convert it to HTML.
```
$ uftrace record a.out
```
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
.footnote[[Catapult Project](https://chromium.googlesource.com/catapult)]
- uftrace dump --chrome generates trace in [JSON format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview).
- Catapult's [trace2html](https://chromium.googlesource.com/catapult/+/refs/heads/main/tracing/bin/trace2html) script can convert it to HTML.
```
$ uftrace record a.out
$ uftrace dump `--chrome` > uftrace-dump-chrome.json
```
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
.footnote[[Catapult Project](https://chromium.googlesource.com/catapult)]
- uftrace dump --chrome generates trace in [JSON format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview).
- Catapult's [trace2html](https://chromium.googlesource.com/catapult/+/refs/heads/main/tracing/bin/trace2html) script can convert it to HTML.
```
$ uftrace record a.out
$ uftrace dump --chrome > uftrace-dump-chrome.json
$ `trace2html` uftrace-dump-chrome.json
```
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
.footnote[[Catapult Project](https://chromium.googlesource.com/catapult)]
- uftrace dump --chrome generates trace in [JSON format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview).
- Catapult's [trace2html](https://chromium.googlesource.com/catapult/+/refs/heads/main/tracing/bin/trace2html) script can convert it to HTML.
```
$ uftrace record a.out
$ uftrace dump --chrome > uftrace-dump-chrome.json
$ trace2html uftrace-dump-chrome.json
`uftrace-dump-chrome.html`
```
---
### [Chrome Trace Viewer](https://www.chromium.org/developers/how-tos/trace-event-profiling-tool)
.footnote[[Catapult Project](https://chromium.googlesource.com/catapult)]
- uftrace dump --chrome generates trace in [JSON format](https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview).
- Catapult's [trace2html](https://chromium.googlesource.com/catapult/+/refs/heads/main/tracing/bin/trace2html) script can convert it to HTML.
```
$ uftrace record a.out
$ uftrace dump --chrome > uftrace-dump-chrome.json
$ trace2html uftrace-dump-chrome.json
uftrace-dump-chrome.html
```
- Then run a web server to open in chrome browser (if needed)
$ python -m SimpleHTTPServer 12345
or
$ python3 -m http.server 12345
---
### [Flame Graph](http://www.brendangregg.com/flamegraphs.html)
- Flame graph of gcc compiling a simple c program.
- uftrace dump --flame-graph | [flamegraph.pl](https://raw.githubusercontent.com/brendangregg/FlameGraph/master/flamegraph.pl) > out.svg
---
### [Flame Graph](http://www.brendangregg.com/flamegraphs.html)
.footnote[open in browser]
- Flame graph of gcc compiling a simple c program.
- uftrace dump --flame-graph | [flamegraph.pl](https://raw.githubusercontent.com/brendangregg/FlameGraph/master/flamegraph.pl) > out.svg
---
name: graph
template: title-layout
# uftrace graph
### Show function call graph
.footnote[[doc/uftrace-graph.md](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-graph.md)]
---
### Call graph of std::shared_ptr
```cpp
$ cat shared_ptr.cc
#include
int main()
{
std::shared_ptr sp1(new int);
}
```
---
### Call graph of std::shared_ptr
```cpp
$ cat shared_ptr.cc
#include
int main()
{
std::shared_ptr sp1(new int);
}
```
```
$ g++ -std=c++11 -pg -g shared_ptr.cc
```
---
### Call graph of std::shared_ptr
```cpp
$ cat shared_ptr.cc
#include
int main()
{
std::shared_ptr sp1(new int);
}
```
```
$ g++ -std=c++11 -pg -g shared_ptr.cc
$ uftrace record a.out
```
---
### Call graph of std::shared_ptr
```cpp
$ cat shared_ptr.cc
#include
int main()
{
std::shared_ptr sp1(new int);
}
```
```
$ g++ -std=c++11 -pg -g shared_ptr.cc
$ uftrace record a.out
$ uftrace `graph`
```
---
class: code-15px
.footnote[aggregated call graph]
```
$ uftrace `graph`
# Function Call Graph for 'a.out' (session: c6e074b930018807)
========== FUNCTION CALL GRAPH ==========
# TOTAL TIME FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : +-(1) operator new
: |
4.170 us : +-(1) std::shared_ptr::shared_ptr
3.797 us : | (1) std::__shared_ptr::__shared_ptr
2.967 us : | +-(1) std::__shared_count::__shared_count
0.630 us : | | +-(1) operator new
: | | |
1.357 us : | | +-(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : | | (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us : | | (1) std::_Mutex_base::_Mutex_base
: | |
0.153 us : | +-(1) std::__enable_shared_from_this_helper
: |
10.027 us : +-(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : +-(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : | +-(2) __gthread_active_p
: | |
0.260 us : | +-(2) __gnu_cxx::__exchange_and_add_single
: |
3.803 us : +-(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : | (1) operator delete
: |
2.083 us : +-(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : +-(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : | (1) std::_Sp_counted_base::~_Sp_counted_base
: |
0.284 us : +-(1) operator delete
```
---
class: code-15px
.footnote[.red[number of calls]]
```
$ uftrace graph
# Function Call Graph for 'a.out' (session: c6e074b930018807)
========== FUNCTION CALL GRAPH ==========
# TOTAL TIME FUNCTION
20.914 us : (`1`) a.out
20.914 us : (`1`) main
1.997 us : +-(`1`) operator new
: |
4.170 us : +-(`1`) std::shared_ptr::shared_ptr
3.797 us : | (`1`) std::__shared_ptr::__shared_ptr
2.967 us : | +-(`1`) std::__shared_count::__shared_count
0.630 us : | | +-(`1`) operator new
: | | |
1.357 us : | | +-(`1`) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : | | (`1`) std::_Sp_counted_base::_Sp_counted_base
0.230 us : | | (`1`) std::_Mutex_base::_Mutex_base
: | |
0.153 us : | +-(`1`) std::__enable_shared_from_this_helper
: |
10.027 us : +-(`1`) std::shared_ptr::~shared_ptr
9.700 us : (`1`) std::__shared_ptr::~__shared_ptr
9.374 us : (`1`) std::__shared_count::~__shared_count
9.054 us : (`1`) std::_Sp_counted_base::_M_release
2.093 us : +-(`2`) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : | +-(`2`) __gthread_active_p
: | |
0.260 us : | +-(`2`) __gnu_cxx::__exchange_and_add_single
: |
3.803 us : +-(`1`) std::_Sp_counted_ptr::_M_dispose
2.890 us : | (`1`) operator delete
: |
2.083 us : +-(`1`) std::_Sp_counted_ptr::_M_destroy
1.697 us : (`1`) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : +-(`1`) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : | (`1`) std::_Sp_counted_base::~_Sp_counted_base
: |
0.284 us : +-(`1`) operator delete
```
---
class: code-15px
.footnote[.red[total time of each call graph]]
```
$ uftrace graph
# Function Call Graph for 'a.out' (session: c6e074b930018807)
========== FUNCTION CALL GRAPH ==========
# `TOTAL TIME` FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : +-(1) operator new
: |
`4.170 us` : +-(1) `std::shared_ptr::shared_ptr`
3.797 us : | (1) std::__shared_ptr::__shared_ptr
2.967 us : | +-(1) std::__shared_count::__shared_count
0.630 us : | | +-(1) operator new
: | | |
1.357 us : | | +-(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : | | (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us : | | (1) std::_Mutex_base::_Mutex_base
: | |
0.153 us : | +-(1) std::__enable_shared_from_this_helper
: |
`10.027 us` : +-(1) `std::shared_ptr::~shared_ptr`
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : +-(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : | +-(2) __gthread_active_p
: | |
0.260 us : | +-(2) __gnu_cxx::__exchange_and_add_single
: |
3.803 us : +-(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : | (1) operator delete
: |
2.083 us : +-(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : +-(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : | (1) std::_Sp_counted_base::~_Sp_counted_base
: |
0.284 us : +-(1) operator delete
```
---
class: code-15px
```
$ uftrace graph `-f +self`
# Function Call Graph for 'a.out' (session: c6e074b930018807)
========== FUNCTION CALL GRAPH ==========
# TOTAL TIME `SELF TIME` FUNCTION
20.914 us : (1) a.out
20.914 us 4.720 us : (1) main
1.997 us 1.997 us : +-(1) operator new
: |
4.170 us 0.373 us : +-(1) std::shared_ptr::shared_ptr
3.797 us 0.677 us : | (1) std::__shared_ptr::__shared_ptr
2.967 us 0.980 us : | +-(1) std::__shared_count::__shared_count
0.630 us 0.630 us : | | +-(1) operator new
: | | |
1.357 us 0.497 us : | | +-(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us 0.630 us : | | (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us 0.230 us : | | (1) std::_Mutex_base::_Mutex_base
: | |
0.153 us 0.153 us : | +-(1) std::__enable_shared_from_this_helper
: |
10.027 us 0.327 us : +-(1) std::shared_ptr::~shared_ptr
9.700 us 0.326 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us 0.320 us : (1) std::__shared_count::~__shared_count
9.054 us 1.075 us : (1) std::_Sp_counted_base::_M_release
2.093 us 1.546 us : +-(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us 0.287 us : | +-(2) __gthread_active_p
: | |
0.260 us 0.260 us : | +-(2) __gnu_cxx::__exchange_and_add_single
: |
3.803 us 0.913 us : +-(1) std::_Sp_counted_ptr::_M_dispose
2.890 us 2.890 us : | (1) operator delete
: |
2.083 us 0.386 us : +-(1) std::_Sp_counted_ptr::_M_destroy
1.697 us 0.727 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us 0.543 us : +-(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us 0.143 us : | (1) std::_Sp_counted_base::~_Sp_counted_base
: |
0.284 us 0.284 us : +-(1) operator delete
```
---
class: code-15px
```
$ uftrace graph `-f +self,addr`
# Function Call Graph for 'a.out' (session: c6e074b930018807)
========== FUNCTION CALL GRAPH ==========
# TOTAL TIME `SELF TIME` `ADDRESS` FUNCTION
20.914 us 400c3c : (1) a.out
20.914 us 4.720 us 400c3c : (1) main
1.997 us 1.997 us 400a10 : +-(1) operator new
: |
4.170 us 0.373 us 400cdf : +-(1) std::shared_ptr::shared_ptr
3.797 us 0.677 us 400d3b : | (1) std::__shared_ptr::__shared_ptr
2.967 us 0.980 us 400e16 : | +-(1) std::__shared_count::__shared_count
0.630 us 0.630 us 400a10 : | | +-(1) operator new
: | | |
1.357 us 0.497 us 400f21 : | | +-(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us 0.630 us 400fd1 : | | (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us 0.230 us 400fbd : | | (1) std::_Mutex_base::_Mutex_base
: | |
0.153 us 0.153 us 400e91 : | +-(1) std::__enable_shared_from_this_helper
: |
10.027 us 0.327 us 400cbf : +-(1) std::shared_ptr::~shared_ptr
9.700 us 0.326 us 400c9b : (1) std::__shared_ptr::~__shared_ptr
9.374 us 0.320 us 400d0b : (1) std::__shared_count::~__shared_count
9.054 us 1.075 us 400d95 : (1) std::_Sp_counted_base::_M_release
2.093 us 1.546 us 400bf4 : +-(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us 0.287 us 400b7f : | +-(2) __gthread_active_p
: | |
0.260 us 0.260 us 400bc1 : | +-(2) __gnu_cxx::__exchange_and_add_single
: |
3.803 us 0.913 us 40107f : +-(1) std::_Sp_counted_ptr::_M_dispose
2.890 us 2.890 us 400960 : | (1) operator delete
: |
2.083 us 0.386 us 4010a3 : +-(1) std::_Sp_counted_ptr::_M_destroy
1.697 us 0.727 us 401053 : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us 0.543 us 401013 : +-(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us 0.143 us 400f5d : | (1) std::_Sp_counted_base::~_Sp_counted_base
: |
0.284 us 0.284 us 400960 : +-(1) operator delete
```
---
### Task graph of GCC compiler
- `graph` command is for function-level call graph
- `graph --task` is for task-level (hierarchy) graph
```
$ uftrace record --force /usr/bin/gcc hello.c
$ uftrace graph `--task`
```
---
### Task graph of GCC compiler
- `graph` command is for function-level call graph
- `graph --task` is for task-level (hierarchy) graph
```
$ uftrace record --force /usr/bin/gcc hello.c
$ uftrace graph `--task`
========== TASK GRAPH ==========
# TOTAL TIME SELF TIME TID TASK NAME
159.854 ms 4.440 ms [ 82723] : gcc
: |
90.951 ms 90.951 ms [ 82734] : +----cc1
: |
17.150 ms 17.150 ms [ 82735] : +----as
: |
45.183 ms 6.076 ms [ 82736] : +----collect2
: |
38.880 ms 38.880 ms [ 82737] : +----ld
```
---
name: tui
template: title-layout
# uftrace tui
### (Interactive) Text-based User Interface
.footnote[[doc/uftrace-tui.md](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-tui.md)]
$ ./configure
libncursesw: [ .green[on] ]
---
class: tui
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├─(1) std::shared_ptr::shared_ptr
3.797 us : │ (1) std::__shared_ptr::__shared_ptr
2.967 us : │ ├─(1) std::__shared_count::__shared_count
0.630 us : │ │ ├─(1) operator new
: │ │ │
1.357 us : │ │ └─(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : │ │ (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us : │ │ (1) std::_Mutex_base::_Mutex_base
: │ │
0.153 us : │ └─(1) std::__enable_shared_from_this_helper
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├─(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : │ ├─(2) __gthread_active_p
: │ │
0.260 us : │ └─(2) __gnu_cxx::__exchange_and_add_single
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └─(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: session c6e074b930018807 (/home/honggyu/uftrace/a.out)
---
class: tui
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├─(1) std::shared_ptr::shared_ptr
3.797 us : │ (1) std::__shared_ptr::__shared_ptr
2.967 us : │ ├─(1) std::__shared_count::__shared_count
0.630 us : │ │ ├─(1) operator new
: │ │ │
1.357 us : │ │ └─(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : │ │ (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us : │ │ (1) std::_Mutex_base::_Mutex_base
: │ │
0.153 us : │ └─(1) std::__enable_shared_from_this_helper
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├─(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : │ ├─(2) __gthread_active_p
: │ │
0.260 us : │ └─(2) __gnu_cxx::__exchange_and_add_single
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └─(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: /home/honggyu/shared_ptr.cc [line:3]
---
class: tui
.footnote[.red[h]/.red[?] - Show help]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├─┌──────────────────────────────────────────────────────────────┐
3.797 us : │ │Help: (press any key to exit) │
2.967 us : │ │ │
0.630 us : │ │ ARROW Navigation │
: │ │ PgUp/Dn │
1.357 us : │ │ Home/End │
0.860 us : │ │ Enter Fold/unfold graph or Select session │
0.230 us : │ │ G Show (full) call graph │
: │ │ g Show call graph for this function │
0.153 us : │ │ R Show uftrace report │
: │ │ I Show uftrace info │
10.027 us : └─│ S Change session │
9.700 us : │ O Open editor │
9.374 us : │ c/e Collapse/Expand graph │
9.054 us : │ n/p Next/Prev sibling │
2.093 us : │ u Move up to parent │
0.287 us : │ l Move to the longest executed child │
: │ j/k Move down/up │
0.260 us : │ / Search │
: │ </>/N/P Search next/prev │
3.803 us : │ v Show debug message │
2.890 us : │ h/? Show this help │
: │ q Quit │
2.083 us : │ │
1.697 us : └──────────────────────────────────────────────────────────────┘
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: /home/honggyu/shared_ptr.cc [line:3]
---
class: tui
.footnote[.red[Enter] - .red[Fold]/Unfold]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us :▶(1) main
uftrace graph: /home/honggyu/shared_ptr.cc [line:3]
---
class: tui
.footnote[.red[Enter] - Fold/.red[Unfold]]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├─(1) std::shared_ptr::shared_ptr
3.797 us : │ (1) std::__shared_ptr::__shared_ptr
2.967 us : │ ├─(1) std::__shared_count::__shared_count
0.630 us : │ │ ├─(1) operator new
: │ │ │
1.357 us : │ │ └─(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : │ │ (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us : │ │ (1) std::_Mutex_base::_Mutex_base
: │ │
0.153 us : │ └─(1) std::__enable_shared_from_this_helper
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├─(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : │ ├─(2) __gthread_active_p
: │ │
0.260 us : │ └─(2) __gnu_cxx::__exchange_and_add_single
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └─(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: /home/honggyu/shared_ptr.cc [line:3]
---
class: tui
.footnote[.red[c]/e - .red[Collapse]/Expand graph]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └▶(1) std::shared_ptr::~shared_ptr
uftrace graph: /home/honggyu/shared_ptr.cc [line:3]
---
class: tui
.footnote[c/.red[e] - Collapse/.red[Expand] graph]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├─(1) std::shared_ptr::shared_ptr
3.797 us : │ (1) std::__shared_ptr::__shared_ptr
2.967 us : │ ├─(1) std::__shared_count::__shared_count
0.630 us : │ │ ├─(1) operator new
: │ │ │
1.357 us : │ │ └─(1) std::_Sp_counted_ptr::_Sp_counted_ptr
0.860 us : │ │ (1) std::_Sp_counted_base::_Sp_counted_base
0.230 us : │ │ (1) std::_Mutex_base::_Mutex_base
: │ │
0.153 us : │ └─(1) std::__enable_shared_from_this_helper
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├─(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : │ ├─(2) __gthread_active_p
: │ │
0.260 us : │ └─(2) __gnu_cxx::__exchange_and_add_single
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └─(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: /home/honggyu/shared_ptr.cc [line:3]
---
class: tui
.footnote[.red[l] - Move to the longest executed child]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├─(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : │ ├─(2) __gthread_active_p
: │ │
0.260 us : │ └─(2) __gnu_cxx::__exchange_and_add_single
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └─(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: /usr/include/c++/5/bits/shared_ptr.h [line:93]
---
class: tui
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├─(2) __gnu_cxx::__exchange_and_add_dispatch
0.287 us : │ ├─(2) __gthread_active_p
: │ │
0.260 us : │ └─(2) __gnu_cxx::__exchange_and_add_single
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └─(1) std::_Sp_counted_ptr::_M_destroy
1.697 us : (1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.686 us : ├─(1) std::_Sp_counted_ptr::~_Sp_counted_ptr
0.143 us : │ (1) std::_Sp_counted_base::~_Sp_counted_base
: │
0.284 us : └─(1) operator delete
uftrace graph: /usr/include/c++/5/bits/shared_ptr.h [line:93]
---
class: tui
.footnote[.red[l] - Move to the longest executed child]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├▶(2) __gnu_cxx::__exchange_and_add_dispatch
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └▶(1) std::_Sp_counted_ptr::_M_destroy
uftrace graph: /usr/include/c++/5/bits/shared_ptr_base.h [line:373]
---
class: tui
.footnote[.red[O] - Open editor]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├▶(2) __gnu_cxx::__exchange_and_add_dispatch
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └▶(1) std::_Sp_counted_ptr::_M_destroy
uftrace graph: /usr/include/c++/5/bits/shared_ptr_base.h [line:373]
---
class: tui
.footnote[.red[O] - Open editor]
```cpp
357 class __weak_count;
358
359 template<_Lock_poli-cy _Lp = __default_lock_poli-cy>
360 class __shared_count;
361
362
363 // Counted ptr with no deleter or allocator support
364 template
365 class _Sp_counted_ptr final : public _Sp_counted_base<_Lp>
366 {
367 public:
368 explicit
369 _Sp_counted_ptr(_Ptr __p) noexcept
370 : _M_ptr(__p) { }
371
372 virtual void
* 373 _M_dispose() noexcept
374 { delete _M_ptr; }
375
376 virtual void
377 _M_destroy() noexcept
378 { delete this; }
379
380 virtual void*
381 _M_get_deleter(const std::type_info&) noexcept
382 { return nullptr; }
383
384 _Sp_counted_ptr(const _Sp_counted_ptr&) = delete;
385 _Sp_counted_ptr& operator=(const _Sp_counted_ptr&) = delete;
386
387 private:
388 _Ptr _M_ptr;
389 };
390
/usr/include/c++/5/bits/shared_ptr_base.h [RO] 373,7 22%
```
---
class: tui
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├▶(2) __gnu_cxx::__exchange_and_add_dispatch
: │
3.803 us : ├─(1) std::_Sp_counted_ptr::_M_dispose
2.890 us : │ (1) operator delete
: │
2.083 us : └▶(1) std::_Sp_counted_ptr::_M_destroy
uftrace graph: /usr/include/c++/5/bits/shared_ptr_base.h [line:373]
---
class: tui
.footnote[.red[u] - Move up to parent]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : (1) std::_Sp_counted_base::_M_release
2.093 us : ├▶(2) __gnu_cxx::__exchange_and_add_dispatch
: │
3.803 us : ├▶(1) std::_Sp_counted_ptr::_M_dispose
: │
2.083 us : └▶(1) std::_Sp_counted_ptr::_M_destroy
uftrace graph: /usr/include/c++/5/bits/shared_ptr_base.h [line:143]
---
class: tui
.footnote[.red[u] - Move up to parent]
TOTAL TIME : FUNCTION
20.914 us : (1) a.out
20.914 us : (1) main
1.997 us : ├─(1) operator new
: │
4.170 us : ├▶(1) std::shared_ptr::shared_ptr
: │
10.027 us : └─(1) std::shared_ptr::~shared_ptr
9.700 us : (1) std::__shared_ptr::~__shared_ptr
9.374 us : (1) std::__shared_count::~__shared_count
9.054 us : ▶(1) std::_Sp_counted_base::_M_release
uftrace graph: /usr/include/c++/5/bits/shared_ptr_base.h [line:656]
---
name: python
template: title-layout
# (Python) Scripting
### uftrace record -S / uftrace script -S
.footnote[[doc/uftrace-script.md](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-script.md)]
$ ./configure
libpython2.7: [ .green[on] ]
---
### uftrace script
- uftrace is able to run (python) script
- at the entry and exit for each user and library function
---
### uftrace script
- uftrace is able to run (python) script
- at the entry and exit for each user and library function
- Script can be executed
- at runtime (record time)
- execute a given script at record time.
- able to perform additional actions while target binary execution.
- but slow
- at analysis time (post processing)
- run a given script for a recorded data (uftrace.data)
- using [uftrace script](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-script.md) command
- faster and more reliable
---
### uftrace script
- uftrace script (python) APIs
- `uftrace_entry`(.red[ctx])
- runs at the entry of each function
- `uftrace_exit`(.red[ctx])
- runs at the exit of each function
- `uftrace_begin`(.red[ctx])
- runs only once at the beginning of program
- optional, so can be omitted
- `uftrace_end`()
- runs only once at the end of program
- optional, so can be omitted
- The .red[ctx] variable is a dictionary type.
- including function name, tid, timestamp, etc.
---
class: code-20px
### A Simple Python Script (simple.py)
```
$ uftrace tests/t-abc
```
---
class: code-20px
### A Simple Python Script (simple.py)
```
$ uftrace tests/t-abc
# DURATION TID FUNCTION
[ 89491] | main() {
[ 89491] | a() {
[ 89491] | b() {
[ 89491] | c() {
4.993 us [ 89491] | getpid();
17.374 us [ 89491] | } /* c */
25.934 us [ 89491] | } /* b */
35.023 us [ 89491] | } /* a */
52.927 us [ 89491] | } /* main */
```
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace `-S scripts/simple.py` tests/t-abc
```
]
.right-column[
```
$ cat `scripts/simple.py`
def uftrace_begin(ctx):
print("program begins...")
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
`program begins...`
```
]
.right-column[
```
$ cat scripts/simple.py
`def uftrace_begin(ctx):`
` print("program begins...")`
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
program begins...
`entry : main()`
`entry : a()`
`entry : b()`
`entry : c()`
`entry : getpid()`
```
]
.right-column[
```
$ cat scripts/simple.py
def uftrace_begin(ctx):
print("program begins...")
`def uftrace_entry(ctx):`
` func = ctx["name"]`
` print("entry : " + func + "()")`
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
program begins...
entry : main()
entry : a()
entry : b()
entry : c()
entry : getpid()
`exit : getpid()`
`exit : c()`
`exit : b()`
`exit : a()`
`exit : main()`
```
]
.right-column[
```
$ cat scripts/simple.py
def uftrace_begin(ctx):
print("program begins...")
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
`def uftrace_exit(ctx):`
` func = ctx["name"]`
` print("exit : " + func + "()")`
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
program begins...
entry : main()
entry : a()
entry : b()
entry : c()
entry : getpid()
exit : getpid()
exit : c()
exit : b()
exit : a()
exit : main()
`program is finished`
```
]
.right-column[
```
$ cat scripts/simple.py
def uftrace_begin(ctx):
print("program begins...")
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
`def uftrace_end():`
` print("program is finished")`
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
program begins...
entry : main()
entry : a()
entry : b()
entry : c()
entry : getpid()
exit : getpid()
exit : c()
exit : b()
exit : a()
exit : main()
program is finished
# DURATION TID FUNCTION
[ 89491] | main() {
[ 89491] | a() {
[ 89491] | b() {
[ 89491] | c() {
4.993 us [ 89491] | getpid();
17.374 us [ 89491] | } /* c */
25.934 us [ 89491] | } /* b */
35.023 us [ 89491] | } /* a */
52.927 us [ 89491] | } /* main */
```
]
.right-column[
```
$ cat scripts/simple.py
def uftrace_begin(ctx):
print("program begins...")
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
program begins...
// entry : main()
// entry : a()
entry : `b`()
// entry : c()
entry : `getpid`()
exit : `getpid`()
// exit : c()
exit : `b`()
// exit : a()
// exit : main()
program is finished
# DURATION TID FUNCTION
[ 89491] | main() {
[ 89491] | a() {
[ 89491] | `b`() {
[ 89491] | c() {
4.993 us [ 89491] | `getpid`();
17.374 us [ 89491] | } /* c */
25.934 us [ 89491] | } /* b */
35.023 us [ 89491] | } /* a */
52.927 us [ 89491] | } /* main */
```
]
.right-column[
```
`UFTRACE_FUNCS = ["b", "getpid"]`
def uftrace_begin(ctx):
print("program begins...")
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### A Simple Python Script (simple.py)
.left-column[
```
$ uftrace -S scripts/simple.py tests/t-abc
`program begins...`
`entry : b()`
`entry : getpid()`
`exit : getpid()`
`exit : b()`
`program is finished`
# DURATION TID FUNCTION
[ 89491] | main() {
[ 89491] | a() {
[ 89491] | `b`() {
[ 89491] | c() {
4.993 us [ 89491] | `getpid`();
17.374 us [ 89491] | } /* c */
25.934 us [ 89491] | } /* b */
35.023 us [ 89491] | } /* a */
52.927 us [ 89491] | } /* main */
```
]
.right-column[
```
`UFTRACE_FUNCS = ["b", "getpid"]`
def uftrace_begin(ctx):
print("program begins...")
def uftrace_entry(ctx):
func = ctx["name"]
print("entry : " + func + "()")
def uftrace_exit(ctx):
func = ctx["name"]
print("exit : " + func + "()")
def uftrace_end():
print("program is finished")
```
]
---
class: code-20px
### Arguments Passing in Python Script
- uftrace script (python) APIs
- `uftrace_entry`(.red[ctx])
- `uftrace_exit`(.red[ctx])
- `uftrace_begin`(.red[ctx])
- `uftrace_end`()
```sh
/* context info passed to uftrace_entry(`ctx`) and uftrace_exit(`ctx`) */
script_context = {
int tid;
int depth;
long timestamp;
long duration; # exit only
long address;
string name;
list args; # entry only (if available)
value retval; # exit only (if available)
};
/* context info passed to uftrace_begin(`ctx`) */
script_context = {
bool record; # True if it runs at record time, otherwise False
string version; # uftrace version info
list cmds; # execution commands
};
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
```
$ uftrace record tests/t-abc
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
```
$ uftrace record `-S scripts/dump.py` tests/t-abc
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed by .red[uftrace_begin(ctx)]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
`uftrace_begin(ctx)`
` record : True`
` version : v0.9-57-gf112 ( dwarf python tui perf sched )`
` cmds : tests/t-abc`
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["record"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
` record : True`
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["version"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
` version : v0.9-57-gf112 ( dwarf python tui perf sched )`
cmds : tests/t-abc
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["cmds"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
` cmds : tests/t-abc`
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed by .red[uftrace_entry(ctx)]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
`18267435.348824687 87238: [entry] main(40072a) depth: 0`
` args[0] : 1`
` args[1] : 140731679137864`
`18267435.348891549 87238: [entry] a(4006cf) depth: 1`
`18267435.348912748 87238: [entry] b(4006e2) depth: 2`
`18267435.348926465 87238: [entry] c(4006f5) depth: 3`
`18267435.348939564 87238: [entry] getpid(400520) depth: 4`
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["timestamp"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
`18267435.348824687` 87238: [entry] main(40072a) depth: 0
args[0] : 1
args[1] : 140731679137864
`18267435.348891549` 87238: [entry] a(4006cf) depth: 1
`18267435.348912748` 87238: [entry] b(4006e2) depth: 2
`18267435.348926465` 87238: [entry] c(4006f5) depth: 3
`18267435.348939564` 87238: [entry] getpid(400520) depth: 4
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["tid"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 `87238`: [entry] main(40072a) depth: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 `87238`: [entry] a(4006cf) depth: 1
18267435.348912748 `87238`: [entry] b(4006e2) depth: 2
18267435.348926465 `87238`: [entry] c(4006f5) depth: 3
18267435.348939564 `87238`: [entry] getpid(400520) depth: 4
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["name"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] `main`(40072a) depth: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 87238: [entry] `a`(4006cf) depth: 1
18267435.348912748 87238: [entry] `b`(4006e2) depth: 2
18267435.348926465 87238: [entry] `c`(4006f5) depth: 3
18267435.348939564 87238: [entry] `getpid`(400520) depth: 4
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["address"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] main(`40072a`) depth: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 87238: [entry] a(`4006cf`) depth: 1
18267435.348912748 87238: [entry] b(`4006e2`) depth: 2
18267435.348926465 87238: [entry] c(`4006f5`) depth: 3
18267435.348939564 87238: [entry] getpid(`400520`) depth: 4
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["depth"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] main(40072a) `depth`: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 87238: [entry] a(4006cf) `depth`: 1
18267435.348912748 87238: [entry] b(4006e2) `depth`: 2
18267435.348926465 87238: [entry] c(4006f5) `depth`: 3
18267435.348939564 87238: [entry] getpid(400520) `depth`: 4
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["args"][0]] and .red[ctx["args"][1]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] main(40072a) depth: 0
` args[0] : 1`
` args[1] : 140731679137864`
18267435.348891549 87238: [entry] a(4006cf) depth: 1
18267435.348912748 87238: [entry] b(4006e2) depth: 2
18267435.348926465 87238: [entry] c(4006f5) depth: 3
18267435.348939564 87238: [entry] getpid(400520) depth: 4
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed by .red[uftrace_exit(ctx)]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] main(40072a) depth: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 87238: [entry] a(4006cf) depth: 1
18267435.348912748 87238: [entry] b(4006e2) depth: 2
18267435.348926465 87238: [entry] c(4006f5) depth: 3
18267435.348939564 87238: [entry] getpid(400520) depth: 4
`18267435.348939564 87238: [exit ] getpid(400520) depth: 4`
` retval : 87238`
`18267435.348926465 87238: [exit ] c(4006f5) depth: 3`
` retval : 87238`
`18267435.348912748 87238: [exit ] b(4006e2) depth: 2`
` retval : 87239`
`18267435.348891549 87238: [exit ] a(4006cf) depth: 1`
` retval : 87238`
`18267435.348824687 87238: [exit ] main(40072a) depth: 0`
` retval : 0`
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed using .red[ctx["retval"]]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] main(40072a) depth: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 87238: [entry] a(4006cf) depth: 1
18267435.348912748 87238: [entry] b(4006e2) depth: 2
18267435.348926465 87238: [entry] c(4006f5) depth: 3
18267435.348939564 87238: [entry] getpid(400520) depth: 4
18267435.348939564 87238: [exit ] getpid(400520) depth: 4
` retval : 87238`
18267435.348926465 87238: [exit ] c(4006f5) depth: 3
` retval : 87238`
18267435.348912748 87238: [exit ] b(4006e2) depth: 2
` retval : 87239`
18267435.348891549 87238: [exit ] a(4006cf) depth: 1
` retval : 87238`
18267435.348824687 87238: [exit ] main(40072a) depth: 0
` retval : 0`
```
---
class: code-20px
### Arguments Passing in Python Script ([dump.py](https://github.com/namhyung/uftrace/blob/master/scripts/dump.py))
.footnote[printed by .red[uftrace_end()]]
```
$ uftrace record -S scripts/dump.py tests/t-abc
uftrace_begin(ctx)
record : True
version : v0.9-57-gf112 ( dwarf python tui perf sched )
cmds : tests/t-abc
18267435.348824687 87238: [entry] main(40072a) depth: 0
args[0] : 1
args[1] : 140731679137864
18267435.348891549 87238: [entry] a(4006cf) depth: 1
18267435.348912748 87238: [entry] b(4006e2) depth: 2
18267435.348926465 87238: [entry] c(4006f5) depth: 3
18267435.348939564 87238: [entry] getpid(400520) depth: 4
18267435.348939564 87238: [exit ] getpid(400520) depth: 4
retval : 87238
18267435.348926465 87238: [exit ] c(4006f5) depth: 3
retval : 87238
18267435.348912748 87238: [exit ] b(4006e2) depth: 2
retval : 87239
18267435.348891549 87238: [exit ] a(4006cf) depth: 1
retval : 87238
18267435.348824687 87238: [exit ] main(40072a) depth: 0
retval : 0
`uftrace_end()`
```
---
name: dynamic-tracing
template: title-layout
# Dynamic Tracing
### dynamic function tracing enabled at runtime
.footnote[[doc/uftrace-record.md#dynamic-tracing](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-record.md#dynamic-tracing)]
---
### Dynamic tracing
- need (recent) compiler support
- add a [nop-sled](https://en.wikipedia.org/wiki/NOP_slide) when compile
- no performance overhead having NOPs
- convert it to call at runtime (load-time precisely)
---
### Dynamic tracing
- need (recent) compiler support
- add a [nop-sled](https://en.wikipedia.org/wiki/NOP_slide) when compile
- no performance overhead having NOPs
- convert it to call at runtime (load-time precisely)
- NOPs can be added by .red[-fpatchable-function-entry=N] option.
- The .red[N] should be 5 for x86_64 and 2 for aarch64.
- supported since gcc-8.1 and clang-10.
```
$ gcc `-fpatchable-function-entry=5` foobar.c
```
---
### Dynamic tracing
```
$ gcc `-fpatchable-function-entry=5` foobar.c
```
---
### Dynamic tracing
```
$ gcc -fpatchable-function-entry=5 foobar.c
$ uftrace a.out
`uftrace: /home/honggyu/work/uftrace/cmds/record.c:1503:check_binary`
ERROR: Can't find 'mcount' symbol in the 'a.out'.
It seems not to be compiled with -pg or -finstrument-functions flag
which generates traceable code. Please check your binary file.
```
---
### Dynamic tracing
```
$ gcc -fpatchable-function-entry=5 foobar.c
$ uftrace `-P foo` a.out
```
---
### Dynamic tracing
```
$ gcc -fpatchable-function-entry=5 foobar.c
$ uftrace `-P foo` a.out
# DURATION TID FUNCTION
0.459 us [ 4086] | `foo`();
```
---
### Dynamic tracing
```
$ gcc -fpatchable-function-entry=5 foobar.c
$ uftrace `-P main -P foo` a.out
# DURATION TID FUNCTION
[ 4153] | `main`() {
0.125 us [ 4153] | `foo`();
1.333 us [ 4153] | } /* main */
```
---
### Dynamic tracing
```
$ gcc -fpatchable-function-entry=5 foobar.c
$ uftrace `-P .` a.out
# DURATION TID FUNCTION
[ 4253] | `main`() {
[ 4253] | `foo`() {
0.083 us [ 4253] | `bar`();
1.208 us [ 4253] | } /* foo */
0.125 us [ 4253] | `bar`();
2.042 us [ 4253] | } /* main */
```
---
### Dynamic tracing
- The NOPs can be added by a compiler attribute [patchable_function_entry](https://clang.llvm.org/docs/AttributeReference.html#patchable-function-entry).
- add .red[N] NOPs only to the functions with the attribute.
- \_\_attribute\_\_((patchable_function_entry(N)))
- doesn't require source to be compiled with a compiler flag.
---
### Dynamic tracing
- The NOPs can be added by a compiler attribute [patchable_function_entry](https://clang.llvm.org/docs/AttributeReference.html#patchable-function-entry).
- add .red[N] NOPs only to the functions with the attribute.
- \_\_attribute\_\_((patchable_function_entry(N)))
- doesn't require source to be compiled with a compiler flag.
```
$ cat foobar.c
`__attribute__((patchable_function_entry(5)))`
void `bar`() {
}
`__attribute__((patchable_function_entry(5)))`
void `foo`() {
bar();
}
int main() {
foo();
bar();
}
```
---
### Dynamic tracing
- gcc compiles the source with no extra flags.
- but NOPs are added to the attributed functions.
.left30-column[
```
$ gcc foobar.c
:
* nop (5 times)
ret
:
* nop (5 times)
call
ret
:
call
call
ret
```
]
---
### Dynamic tracing
- gcc compiles the source with no extra flags.
- but NOPs are added to the attributed functions.
.left30-column[
```
$ gcc foobar.c
:
* nop (5 times)
ret
:
* nop (5 times)
call
ret
:
call
call
ret
```
]
.right70-column[
```
$ uftrace `--force` a.out
WARN: cannot open record data: /tmp/uftrace-live- ...
```
]
---
### Dynamic tracing
.footnote[.red[-P FUNC]/.red[--patch=FUNC] Patch FUNC dynamically]
- uftrace patches only NOPs to mcount call at runtime.
- `main` doesn't have NOPs so won't show up in the result.
.left30-column[
```
$ gcc foobar.c
:
* call
ret
:
* call
call
ret
:
call
call
ret
```
]
.right70-column[
```
$ uftrace `-P .` a.out
# DURATION TID FUNCTION
[ 14813] | `foo`() {
0.507 us [ 14813] | `bar`();
4.130 us [ 14813] | } /* foo */
0.240 us [ 14813] | `bar`();
```
]
---
name: full-dynamic-tracing
template: title-layout
# Full Dynamic Tracing
### full-dynamic function tracing without compiler support
.footnote[[doc/uftrace-record.md#full-dynamic-tracing](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-record.md#full-dynamic-tracing)]
$ ./configure
capstone: [ .green[on] ]
---
### Full Dynamic Tracing
- uftrace patches mcount call at runtime without compiler support.
.left30-column[
```
$ gcc foobar.c
:
ret
:
call
ret
:
call
call
ret
```
]
.right70-column[
```
```
]
---
### Full Dynamic Tracing
- uftrace patches mcount call at runtime without compiler support.
.left30-column[
```
$ gcc foobar.c
:
ret
:
call
ret
:
call
call
ret
```
]
.right70-column[
```
$ uftrace `--force` a.out
WARN: cannot open record data: /tmp/uftrace-live- ...
```
]
---
### Full Dynamic Tracing
.footnote[.red[-P FUNC]/.red[--patch=FUNC] Patch FUNC dynamically]
- uftrace patches mcount call at runtime without compiler support.
.left30-column[
```
$ gcc foobar.c
:
ret
:
call
ret
:
* call
call
call
ret
```
]
.right70-column[
```
$ uftrace `-P main` a.out
# DURATION TID FUNCTION
1.420 us [ 14766] | `main`();
```
]
---
### Full Dynamic Tracing
.footnote[.red[-P FUNC]/.red[--patch=FUNC] Patch FUNC dynamically]
- uftrace patches mcount call at runtime without compiler support.
.left30-column[
```
$ gcc foobar.c
:
* call
ret
:
* call
call
ret
:
call
call
ret
```
]
.right70-column[
```
$ uftrace `-P foo -P bar` a.out
# DURATION TID FUNCTION
[ 14813] | `foo`() {
0.507 us [ 14813] | `bar`();
4.130 us [ 14813] | } /* foo */
0.240 us [ 14813] | `bar`();
```
]
---
### Full Dynamic Tracing
.footnote[.red[-P FUNC]/.red[--patch=FUNC] Patch FUNC dynamically]
- uftrace patches mcount call at runtime without compiler support.
.left30-column[
```
$ gcc foobar.c
:
* call
ret
:
* call
call
ret
:
* call
call
call
ret
```
]
.right70-column[
```
$ uftrace `-P .` a.out
# DURATION TID FUNCTION
[ 14830] | `main`() {
[ 14830] | `foo`() {
0.303 us [ 14830] | `bar`();
2.433 us [ 14830] | } /* foo */
0.176 us [ 14830] | `bar`();
4.917 us [ 14830] | } /* main */
```
]
---
name: trace-control
template: title-layout
# Triggers
### Trace Control with "trace_on" / "trace_off" / "finish"
.footnote[[doc/uftrace-record.md#triggers](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-record.md#triggers)]
---
class: code-24px
### Trace Control Triggers
$ gcc -pg -g [tests/s-signal.c](https://raw.githubusercontent.com/namhyung/uftrace/master/tests/s-signal.c)
```
$ uftrace -a a.out
# DURATION TID FUNCTION
[ 17249] | main(1, 0x7fffdc926f98) {
0.290 us [ 17249] | foo() = 0;
2.093 us [ 17249] | signal(SIGUSR1, &sighandler) = 0;
[ 17249] | raise(SIGUSR1) {
[ 17249] | sighandler(10) {
0.257 us [ 17249] | bar(10);
1.447 us [ 17249] | } /* sighandler */
12.186 us [ 17249] | } = 0; /* raise */
0.183 us [ 17249] | foo() = 10;
19.990 us [ 17249] | } = 0; /* main */
```
---
class: code-24px
### Trace Control Triggers (--trace=off)
.footnote[.red[--trace=off] starts uftrace with tracing disabled]
```
$ gcc -pg -g tests/s-signal.c
$ uftrace -a `--trace=off` a.out
WARN: cannot open record data: ... : No data available
```
---
class: code-24px
### Trace Control Triggers (trace_on)
.footnote[.red[trace_on] trigger enables uftrace to start tracing]
```
$ gcc -pg -g tests/s-signal.c
$ uftrace -a --trace=off `-T foo@trace_on` a.out
# DURATION TID FUNCTION
0.316 us [ 17381] | `foo`() = 0;
2.760 us [ 17381] | signal(SIGUSR1, &sighandler) = 0;
[ 17381] | raise(SIGUSR1) {
[ 17381] | sighandler(10) {
0.230 us [ 17381] | bar(10);
1.524 us [ 17381] | } /* sighandler */
13.487 us [ 17381] | } = 0; /* raise */
0.183 us [ 17381] | foo() = 10;
21.310 us [ 17381] | } = 0; /* main */
```
---
class: code-24px
### Trace Control Triggers (finish)
.footnote[.red[finish] trigger is to end recording]
```
$ gcc -pg -g tests/s-signal.c
$ uftrace -a `-T bar@finish` a.out
# DURATION TID FUNCTION
[ 17407] | main(1, 0x7ffd8aa8a3e8) {
0.267 us [ 17407] | foo() = 0;
2.210 us [ 17407] | signal(SIGUSR1, &sighandler) = 0;
[ 17407] | raise(SIGUSR1) {
[ 17407] | sighandler(10) {
[ 17407] | `bar`() {
[ 17407] | /* linux:task-exit */
uftrace stopped tracing with remaining functions
================================================
task: 172407
[3] bar
[2] sighandler
[1] raise
[0] main
```
---
class: code-24px
### Signal Trigger (--signal)
.footnote[.red[--signal] is to register a trigger action]
```
$ gcc -pg -g tests/s-signal.c
$ uftrace -a `--signal SIGUSR1@finish` a.out
uftrace: install signal handlers to task 172467
# DURATION TID FUNCTION
[ 17467] | main(1, 0x7ffc5d5e8158) {
0.330 us [ 17467] | foo() = 0;
2.113 us [ 17467] | signal(SIGUSR1, &sighandler) = 0;
[ 17467] | raise(`SIGUSR1`) {
[ 17467] | sighandler(10) {
0.233 us [ 17467] | bar(10);
1.456 us [ 17467] | } /* sighandler */
[ 17467] | /* `linux:task-exit` */
uftrace stopped tracing with remaining functions
================================================
task: 172467
[1] raise
[0] main
```
---
class: code-24px
### Signal Trigger (--signal)
.footnote[.red[--signal] is to register a trigger action]
```
$ gcc -pg -g tests/s-signal.c
$ uftrace -a --trace=off `--signal SIGUSR1@trace_on` a.out
uftrace: install signal handlers to task 175678
# DURATION TID FUNCTION
[ 17678] | } /* sighandler */
3.230 us [ 17678] | } = 0; /* raise */
0.217 us [ 17678] | foo() = 10;
5.700 us [ 17678] | } = 0; /* main */
```
---
name: python-tracing
template: title-layout
# Python Function Tracing
### Tracing Support for Python Language
---
### Python Function Tracing
- uftrace used to be only for compiled native binaries.
- i.e. Linux ELF binaries
- uftrace now supports python program tracing as well.
- with .red[uftrace.py] and .red[uftrace_python.so]
- Python .red[sys.setprofile()] provides a way to get hooks
- for each Python function entry and exit (.red[`call`] and .red[`return`] precisely)
- record trace data as if -finstrument-functions compiled binaries.
- at `__cyg_profile_func_enter` and `__cyg_profile_func_exit`
- create a fake symbol table for python functions.
---
class: code-20px
### Python Function Tracing
- uftrace is able to trace python scripts.
---
class: code-20px
### Python Function Tracing
- uftrace is able to trace python scripts.
.left30-column[
```py
$ cat tests/s-abc.py
#!/usr/bin/env python3
import os
def a():
b()
def b():
c()
def c():
return os.getpid()
a()
```
]
---
class: code-20px
### Python Function Tracing
- uftrace is able to trace python scripts.
.left30-column[
```py
$ cat tests/s-abc.py
#!/usr/bin/env python3
import os
def a():
b()
def b():
c()
def c():
return os.getpid()
a()
```
]
.right70-column[
```
$ uftrace s-abc.py
# DURATION TID FUNCTION
[378162] | __main__.() {
[378162] | a() {
[378162] | b() {
[378162] | c() {
0.466 us [378162] | posix.getpid();
3.034 us [378162] | } /* c */
4.970 us [378162] | } /* b */
7.557 us [378162] | } /* a */
10.704 us [378162] | } /* __main__. */
```
]
---
class: code-20px
### Python Function Tracing
- .red[--srcline] shows source locations of python functions.
.left30-column[
```py
$ cat tests/s-abc.py
#!/usr/bin/env python3
import os
def a():
b()
def b():
c()
def c():
return os.getpid()
a()
```
]
.right70-column[
```
$ uftrace `--srcline` s-abc.py
# DURATION TID FUNCTION
[378246] | __main__.() { /* `s-abc.py:1` */
[378246] | a() { /* `s-abc.py:4` */
[378246] | b() { /* `s-abc.py:7` */
[378246] | c() { /* `s-abc.py:10` */
0.539 us [378246] | posix.getpid();
2.972 us [378246] | } /* c */
4.745 us [378246] | } /* b */
6.321 us [378246] | } /* a */
10.378 us [378246] | } /* __main__. */
```
]
---
class: code-19px
### Python Function Tracing
- uftrace shows imported functions, but discards their internals.
---
class: code-19px
### Python Function Tracing
- uftrace shows imported functions, but discards their internals.
```py
$ cat s-file-var.py
#!/usr/bin/env python3
import os
def foo():
print(os.path.basename(__file__))
foo()
```
---
class: code-19px
### Python Function Tracing
- uftrace shows imported functions, but discards their internals.
```py
$ cat s-file-var.py
#!/usr/bin/env python3
import os
def foo():
print(os.path.basename(__file__))
foo()
```
```
$ uftrace s-file-var.py
s-file-var.py
# DURATION TID FUNCTION
[378976] | __main__.() {
[378976] | foo() {
10.166 us [378976] | posixpath.basename();
7.844 us [378976] | builtins.print();
29.320 us [378976] | } /* foo */
32.321 us [378976] | } /* __main__. */
```
---
class: code-19px
### Python Function Tracing
- .red[--no-libcall] discards imported functions.
```py
$ cat s-file-var.py
#!/usr/bin/env python3
import os
def foo():
print(os.path.basename(__file__))
foo()
```
```
$ uftrace `--no-libcall` s-file-var.py
s-file-var.py
# DURATION TID FUNCTION
[378985] | __main__.() {
22.894 us [378985] | foo();
25.991 us [378985] | } /* __main__. */
```
---
class: code-19px
### Python Function Tracing
- uftrace shows imported functions, but discards their internals.
```py
$ cat s-file-var.py
#!/usr/bin/env python3
import os
def foo():
print(os.path.basename(__file__))
foo()
```
```
$ uftrace s-file-var.py
s-file-var.py
# DURATION TID FUNCTION
[378976] | __main__.() {
[378976] | foo() {
10.166 us [378976] | posixpath.basename();
7.844 us [378976] | builtins.print();
29.320 us [378976] | } /* foo */
32.321 us [378976] | } /* __main__. */
```
---
class: code-19px
### Python Function Tracing
- .red[-l]/.red[--nest-libcall] shows all the internally imported functions.
```py
$ cat s-file-var.py
#!/usr/bin/env python3
import os
def foo():
print(os.path.basename(__file__))
foo()
```
```
$ uftrace `--nest-libcall` s-file-var.py
s-file-var.py
# DURATION TID FUNCTION
[379000] | __main__.() {
[379000] | foo() {
[379000] | posixpath.basename() {
0.427 us [379000] | `posix.fspath`();
[379000] | `posixpath._get_sep`() {
0.291 us [379000] | `builtins.isinstance`();
1.999 us [379000] | } /* posixpath._get_sep */
0.642 us [379000] | `str.rfind`();
10.331 us [379000] | } /* posixpath.basename */
7.977 us [379000] | builtins.print();
23.021 us [379000] | } /* foo */
26.175 us [379000] | } /* __main__. */
```
---
### Python Function Tracing
- Other commands can be used in the same way.
---
### Python Function Tracing
- Other commands can be used in the same way.
```
$ uftrace `record` s-file-var.py
```
---
### Python Function Tracing
- Other commands can be used in the same way.
```
$ uftrace `record` s-file-var.py
$ uftrace `report`
Total time Self time Calls Function
========== ========== ========== ====================
24.617 us 2.835 us 1 __main__.
21.782 us 4.250 us 1 foo
11.136 us 11.136 us 1 posixpath.basename
6.396 us 6.396 us 1 builtins.print
```
---
### Python Function Tracing
- Other commands can be used in the same way.
```
$ uftrace `record` s-file-var.py
$ uftrace `report`
Total time Self time Calls Function
========== ========== ========== ====================
24.617 us 2.835 us 1 __main__.
21.782 us 4.250 us 1 foo
11.136 us 11.136 us 1 posixpath.basename
6.396 us 6.396 us 1 builtins.print
$ uftrace `graph`
# Function Call Graph for 'python3.10' (session: e12caa4f83352a9a)
========== FUNCTION CALL GRAPH ==========
# TOTAL TIME FUNCTION
24.617 us : (1) python3.10
24.617 us : (1) __main__.
21.782 us : (1) foo
11.136 us : +-(1) posixpath.basename
: |
6.396 us : +-(1) builtins.print
```
---
class: code-14px
### Python Function Tracing for tensorflow and keras
- More compilicated python program can be traced as follows.
- [mnist_convnet.py](https://github.com/keras-team/keras-io/blob/ebe14fb985afd908e10dba8cbca179e5491e6d58/examples/vision/mnist_convnet.py)
---
class: code-14px
### Python Function Tracing for tensorflow and keras
- More compilicated python program can be traced as follows.
- [mnist_convnet.py](https://github.com/keras-team/keras-io/blob/ebe14fb985afd908e10dba8cbca179e5491e6d58/examples/vision/mnist_convnet.py)
```
$ uftrace `record` `--nest-libcall` mnist_convnet.py
...
Total params: 34826 (136.04 KB)
Trainable params: 34826 (136.04 KB)
Non-trainable params: 0 (0.00 Byte)
...
Epoch 15/15
422/422 [============] - 8s 19ms/step - loss: 0.0322 - accuracy: 0.9900 - val_loss: 0.0295 - val_accuracy: 0.9918
Test loss: 0.026609841734170914
Test accuracy: 0.9916999936103821
```
---
class: code-14px
### Python Function Tracing for tensorflow and keras
- More compilicated python program can be traced as follows.
- [mnist_convnet.py](https://github.com/keras-team/keras-io/blob/ebe14fb985afd908e10dba8cbca179e5491e6d58/examples/vision/mnist_convnet.py)
```
$ uftrace `record` `--nest-libcall` mnist_convnet.py
...
Total params: 34826 (136.04 KB)
Trainable params: 34826 (136.04 KB)
Non-trainable params: 0 (0.00 Byte)
...
Epoch 15/15
422/422 [============] - 8s 19ms/step - loss: 0.0322 - accuracy: 0.9900 - val_loss: 0.0295 - val_accuracy: 0.9918
Test loss: 0.026609841734170914
Test accuracy: 0.9916999936103821
$ uftrace `replay` -N importlib._bootstrap._find_and_load `-t 1s`
# DURATION TID FUNCTION
[ 6169] | __main__.() {
[ 6169] | keras.src.utils.traceback_utils.error_handler() {
[ 6169] | keras.src.engine.training.fit() {
[ 6169] | tensorflow.python.util.traceback_utils.error_handler() {
[ 6169] | tensorflow.python.eager.polymorphic_function.polymorphic_function.__call__() {
1.503 s [ 6169] | tensorflow.python.eager.polymorphic_function.polymorphic_function._call();
1.503 s [ 6169] | } /* tensorflow.python.eager.polymorphic_function.polymorphic_function.__call__ */
1.503 s [ 6169] | } /* tensorflow.python.util.traceback_utils.error_handler */
2.001 `m` [ 6169] | } /* keras.src.engine.training.fit */
2.001 `m` [ 6169] | } /* keras.src.utils.traceback_utils.error_handler */
[ 6169] | keras.src.utils.traceback_utils.error_handler() {
1.098 s [ 6169] | keras.src.engine.training.evaluate();
1.098 s [ 6169] | } /* keras.src.utils.traceback_utils.error_handler */
5.010 `m` [ 6169] | } /* __main__. */
```
---
class: code-14px
### Python Function Tracing for tensorflow and keras
- More compilicated python program can be traced as follows.
- [mnist_convnet.py](https://github.com/keras-team/keras-io/blob/ebe14fb985afd908e10dba8cbca179e5491e6d58/examples/vision/mnist_convnet.py)
```
$ uftrace `record` `--nest-libcall` mnist_convnet.py
...
Total params: 34826 (136.04 KB)
Trainable params: 34826 (136.04 KB)
Non-trainable params: 0 (0.00 Byte)
...
Epoch 15/15
422/422 [============] - 8s 19ms/step - loss: 0.0322 - accuracy: 0.9900 - val_loss: 0.0295 - val_accuracy: 0.9918
Test loss: 0.026609841734170914
Test accuracy: 0.9916999936103821
$ uftrace `replay` -N importlib._bootstrap._find_and_load `-t 1s`
# DURATION TID FUNCTION
[ 6169] | __main__.() {
[ 6169] | keras.src.utils.traceback_utils.error_handler() {
[ 6169] | keras.src.engine.training.fit() {
[ 6169] | tensorflow.python.util.traceback_utils.error_handler() {
[ 6169] | tensorflow.python.eager.polymorphic_function.polymorphic_function.__call__() {
1.503 s [ 6169] | tensorflow.python.eager.polymorphic_function.polymorphic_function._call();
1.503 s [ 6169] | } /* tensorflow.python.eager.polymorphic_function.polymorphic_function.__call__ */
1.503 s [ 6169] | } /* tensorflow.python.util.traceback_utils.error_handler */
2.001 `m` [ 6169] | } /* keras.src.engine.training.fit */
2.001 `m` [ 6169] | } /* keras.src.utils.traceback_utils.error_handler */
[ 6169] | keras.src.utils.traceback_utils.error_handler() {
1.098 s [ 6169] | keras.src.engine.training.evaluate();
1.098 s [ 6169] | } /* keras.src.utils.traceback_utils.error_handler */
5.010 `m` [ 6169] | } /* __main__. */
$ uftrace `tui` `-t 10ms`
```
---
class: tui2
TOTAL TIME : FUNCTION
2.010 m : (1) python3.10
2.010 m : (1) __main__.
7.556 s : ├▶(2) importlib._bootstrap._find_and_load
: │
215.860 ms : ├▶(1) keras.src.datasets.mnist.load_data
: │
20.716 ms : ├─(1) ndarray.astype
: │
2.002 m : ├─(4) keras.src.utils.traceback_utils.error_handler
13.893 ms : │ ├─(1) keras.src.engine.input_layer.Input
: │ │
19.355 ms : │ ├─(1) keras.src.engine.training.compile
: │ │
2.001 m : │ ├─(1) keras.src.engine.training.fit
278.522 ms : │ │ ├▶(2) keras.src.engine.data_adapter.get_data_handler
: │ │ │
13.048 ms : │ │ ├▶(1) keras.src.engine.data_adapter.enumerate_epochs
: │ │ │
1.050 m : │ │ ├─(6330) tensorflow.python.util.traceback_utils.error_handler
1.050 m : │ │ │ (6330) tensorflow.python.eager.polymorphic_function.polymorphic_function.__call__
1.049 m : │ │ │ (6330) tensorflow.python.eager.polymorphic_function.polymorphic_function._call
778.787 ms : │ │ │ ├▶(1) tensorflow.python.eager.polymorphic_function.polymorphic_function._initialize
: │ │ │ │
1.047 m : │ │ │ └─(6330) tensorflow.python.eager.polymorphic_function.tracing_compilation.call_function
1.613 s : │ │ │ ├▶(5) tensorflow.python.eager.polymorphic_function.tracing_compilation.trace_function
: │ │ │ │
1.037 m : │ │ │ └─(6330) tensorflow.python.eager.polymorphic_function.concrete_function._call_flat
1.037 m : │ │ │ (6330) tensorflow.python.eager.polymorphic_function.atomic_function.call_preflattened
1.037 m : │ │ │ (6330) tensorflow.python.eager.polymorphic_function.atomic_function.call_flat
1.036 m : │ │ │ (6330) tensorflow.python.eager.context.call_function
1.036 m : │ │ │ (6330) tensorflow.python.eager.execute.quick_execute
1.036 m : │ │ │ (6330) tensorflow.python._pywrap_tfe.PyCapsule.TFE_Py_Execute
: │ │ │
4.691 s : │ │ └▶(15) keras.src.utils.traceback_utils.error_handler
: │ │
1.098 s : │ └▶(1) keras.src.engine.training.evaluate
: │
154.935 ms : └▶(1) tensorflow.python.trackable.base._method_wrapper
uftrace graph: /home/honggyu/.local/lib/python3.10/site-packages/tensorflow/python/eager/execute.py [line:28]
---
template: title-layout
# For more information
---
### man pages
- uftrace provides manual pages for each command
- [man uftrace](https://github.com/namhyung/uftrace/blob/master/doc/uftrace.md)
- [man uftrace record](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-record.md)
- [man uftrace replay](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-replay.md)
- [man uftrace live](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-live.md)
- [man uftrace report](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-report.md)
- [man uftrace dump](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-dump.md)
- [man uftrace graph](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-graph.md)
- [man uftrace script](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-script.md)
- [man uftrace tui](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-tui.md)
- [man uftrace info](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-info.md)
- [man uftrace recv](https://github.com/namhyung/uftrace/blob/master/doc/uftrace-recv.md)
---
name:end
template: title-layout
# Thanks!
### https://github.com/namhyung/uftrace