Debug and profile
In this section, you will learn:
How to debug TiDB
How to pause the execution at any line of code to inspect values and stacks
How to profile TiDB to catch a performance bottleneck
Use delve for debugging
Delve is a debugger for the Go programming language. It provides a command-line debugging experience similar to the GNU Project debugger (GDB), but it is much more Go native than GDB itself.
Install delve
To install delve, see the installation guide. After the installation, depending on how you set your environment variables, you will have an executable file named dlv
in either $GOPATH/bin
or $HOME/go/bin
. You can then run the following command to verify the installation:
Attach delve to a running TiDB process
Once you get the TiDB server running, you can attach the delve debugger.
For example, you can build and run a standalone TiDB server by running the following commands in the root directory of the source code:
You can then start a new shell and use ps
or pgrep
to find the PID of the tidb server process you just started:
If the output lists multiple PIDs, it indicates that you might have multiple TiDB servers running at the same time. To determine the PID of the tidb server you are planning to debug, you can use commands such as ps $PID
, where $PID
is the PID you are trying to know more about:
Once you get the PID, you can attach delve to it by running the following command:
You might get error messages of the kernel security setting as follows:
To resolve the error, follow the instructions provided in the error message and execute the following command as the root user to override the kernel security setting:
Then retry attaching delve onto the PID, and it should work.
If you've worked with GDB, the delve debugging interface will look familiar to you. It is an interactive dialogue that allows you to interact with the execution of the tidb server attached on. To learn more about delve, you can type help into the dialogue and read the help
messages.
Use delve for debugging
After attaching delve to the running TiDB server process, you can now set breakpoints. TiDB server will pause execution at the breakpoints you specify.
To create a breakpoint, you can write:
where [name]
is the name for the breakpoint, and <linespec>
is the position of a line of code in the source code. Note the name is optional.
For example, the following command creates a breakpoint at the Next
function of HashJoinExec
. (The line number can be subject to change due to the modification of the source code).
Once the execution is paused, the context of the execution is fully preserved. You are free to inspect the values of different variables, print the calling stack, and even jump between different goroutines. Once you finish the inspection, you can resume the execution by stepping into the next line of code or continue the execution until the next breakpoint is encountered.
Typically, when you use a debugger, you need to take the following steps:
Locate the code and set a breakpoint.
Prepare data so that the execution will get through the breakpoint, and pause at the specified breakpoint as expected.
Inspect values and follow the execution step by step.
Using delve to debug a test case
If a test case fails, you can also use delve to debug it. Get the name of the test case, go to the corresponding package directory, and then run the following command to start a debugging session that will stop at the entry of the test:
Understand how TiDB works through debugging
Besides debugging problems, you can also use the debugger to understand how TiDB works through tracking the execution step by step.
To understand TiDB internals, it's critical that you understand certain functions. To better understand how TiDB works, you can pause the execution of these TiDB functions, and then run TiDB step by step.
For example:
executor/compiler.go:Compile
is where each SQL statement is compiled and optimized.planner/planner.go:Optimize
is where the SQL optimization starts.executor/adapter.go:ExecStmt.Exec
is where the SQL plan turns into executor and where the SQL execution starts.Each executor's
Open
,Next
, andClose
function marks the volcano-style execution logic.
When you are reading the TiDB source code, you are strongly encouraged to set a breakpoint and use the debugger to trace the execution whenever you are confused or uncertain about the code.
Using pprof
for profiling
pprof
for profilingFor any database system, performance is always important. If you want to know where the performance bottleneck is, you can use a powerful Go profiling tool called pprof
.
Gather runtime profiling information through HTTP end points
Usually, when TiDB server is running, it exposes a profiling end point through HTTP at http://127.0.0.1:10080/debug/pprof/profile
. You can get the profile result by running the following commands:
The commands capture the profiling information for 45 seconds, and then provide a web view of the profiling result at 127.0.0.1:4001
. This view contains a flame graph of the execution and more views that can help you diagnose the performance bottleneck.
You can also gather other runtime information through this end point. For example:
Goroutine:
Trace:
Heap:
To learn how the runtime information is analyzed, see Go's diagnostics document.
Profiling during benchmarking
When you are proposing a performance-related feature for TiDB, we recommend that you also include a benchmark result as proof of the performance gain or to show that your code won't introduce any performance regression. In this case, you need to write your own benchmark test like in executor/benchmark.go
.
For example, if you want to benchmark the window functions, because BenchmarkWindow
are already in the benchmark tests, you can run the following commands to get the benchmark result:
If you find any performance regression, and you want to know the cause of it, you could use a command like the following:
Then, you can use the steps described above to generate and analyze the profiling information.
Last updated