As our previous post described, TiDB 4.0 is an easier-to-use, production-ready Hybrid Transactional/Analytical Processing (HTAP) database. Our long-term mission is to build a high-performance distributed SQL database to help you solve your business growth pains in Online Transactional Processing (OLTP) and real-time analytical scenarios.
Transaction Processing Performance Council (TPC) benchmarks, which measure database system performance, are accepted gold standards in the industry. To test the performance of our upcoming TiDB 4.0 release, we used TPC-C specifications to measure its OLTP capabilities and TPC-H specifications to measure its Online Analytical Processing (OLAP) capabilities.
Our test results showed that, compared with version 3.0, TiDB 4.0’s TPC-C performance improved by about 50%, and TPC-H performance increased by about 100% on average over a set of 22 queries.
In this post, we’ll show you how we significantly enhanced TiDB 4.0’s TPC-C and TPC-H performance.
TPC-C performance improved by 50%
We ran a TPC-C test against the TiDB 4.0 general availability (GA) release and TiDB 3.0.13, and we found that the TiDB 4.0 performed about 50% better.
Hardware configurations:
Component | Instances | vCPUs | RAM (GB) | Instance type |
TiKV | 3 | 16 | 122 | AWS i3.4xlarge |
TiDB | 3 | 16 | 32 | AWS c5.4xlarge |
Placement Driver (PD) | 3 | 4 | 16 | AWS m5.xlarge |
For configuration parameters and test steps, see this document.
Here are the TPC-C test results:
We achieved this result because we optimized TiDB’s pessimistic transactions, the Raft thread model, and RocksDB’s write throughput.
Optimizing pessimistic transactions
As our previous post described, we’ve improved pessimistic locking’s performance, stability, and compatibility with MySQL. Pessimistic locking becomes generally available in TiDB 4.0. In this section, we’ll introduce how we reduce remote procedure calls (RPCs) and latency of client requests.
Lock requests return the required data
In TiDB 3.0, when you execute an SQL statement like “UPDATE column1 =? WHERE id =?” in a pessimistic transaction, TiDB must write the pessimistic lock label in the row’s lock column family and then query the row’s data. In this process, there are two RPC interactions between TiDB and TiKV.
But in 4.0, we merge these two operations into a single step. When TiKV performs the lock operation, it returns the data you require to TiDB. You can see the code here.
Lock requests immediately return data to the client
When we analyzed the TPC-C client’s execution time, we found that the client spent most of its time waiting for the pessimistic lock request to complete. In a TPC-C test, a transaction modified multiple rows, and each row required a pessimistic lock. Besides, TiKV replicated data among multiple replicas. Therefore, write request latency was much higher than read request latency. Because the client’s execution latency was too high, it was difficult to further improve TiKV’s resource utilization and the client’s queries per second (QPS).
After analysis, we found that the pessimistic lock request was different from other write requests: the pessimistic lock request could fail. In the two-phase commit protocol’s commit phase, if the pessimistic lock label that should exist was overwritten by another transaction or another transaction had written a newer version, this transaction rolled back.
Here is the locking process before optimization:
Now, after we optimized TiKV’s execution logic, if TiKV determines that there isn’t a lock conflict, it doesn’t wait for Raftstore to replicate logs to multiple replicas. Instead, it immediately returns data to the client and assumes that the lock is successfully applied. This greatly reduces the latency of client requests. You can see the code here.
Here is the locking process after optimization:
Optimizing Raftstore’s write model
Before we discuss our optimization on the Raft thread model, let’s give you a bit of background knowledge about Raftstore. Our previous post mentioned that the TiKV cluster implements the Raft consensus algorithm to ensure data consistency. In a Raft group, the leader replicates an entry to all followers to keep logs consistent. When the entry is replicated to the majority of peers in a Raft group, it means that it is successfully written to TiKV. As another post described, TiKV aims to support 100 TB+ of data, but that’s impossible for one Raft group to achieve. We need to use multiple Raft groups, which is Multi-raft. In TiKV, the implementation of Multi-raft is completed in Raftstore. You can find the code in the raftstore/store directory.
In TiDB 3.0, when TiKV wrote data to Raftstore, it needed to successfully replicate logs to the majority of followers, write the committed index that was agreed upon by consensus to raftdb
, and persist the data. Then, TiKV could start to apply the data to kvdb
that was for your access (the apply operation).
However, according to the Raft paper[1], the committed index does not need to be persistent. As long as the log is agreed by the majority of replicas, this piece of data will not be lost. Therefore, in TiDB 4.0, we advance the apply operation before committed index persistence, so that the written data can be more quickly returned to TiDB. You can see the code here.
Optimizing RocksDB’s write throughput
As TiKV’s underlying key-value engine, RocksDB provides TiKV with snapshot-isolation-level reads and writes. When multiple threads are writing to RocksDB, to ensure linearizability of reads and writes, RocksDB must sort the write requests of multiple threads. After the sorting is completed, a single thread writes to the write-ahead log (WAL) to ensure that the data is not lost. Then, multiple threads write to memtable
in parallel, and the global version number is updated so that you can read these requests.
However, if a thread in RocksDB is executing the request of inserting to memtable
, a new thread must wait for all write requests to complete before inserting data again. This greatly limits RocksDB’s multi-thread throughput. Although RocksDB provides the unordered write feature, in which multiple threads don’t need to queue or wait, this destroys snapshot isolation. It requires the upper-layer application to use other designs to guarantee program correctness. This increases application complexity.
Inspired by the pipeline model in the paper X-Engine: An Optimized Storage Engine for Large-scale E-commerce Transaction Processing[2], we split a writebatch that is written to memtable
into multiple subtasks and put them in the task pool. If any thread that tries to write to RocksDB is blocked in a certain step, it can go to the task pool to execute tasks. This approach uses threads more efficiently, and idle threads can help the main thread share the write operation. For the code, see pull request (PR) #7111 in tikv, PR #131 in tikv/rocksdb, and PR #154 in tikv/rocksdb.
TPC-H performance improved by 100%
We ran TPC-H queries on TiDB 3.0.13 and TiDB 4.0 to compare their OLAP capabilities, and we found that TiDB’s query performance notably improved—about 100% on average.
Because TiDB 4.0 introduces TiFlash to strengthen TiDB’s HTAP capabilities, our test objects were:
- TiDB 3.0.13 that only read data from TiKV
- TiDB 4.0 that only read data from TiKV
- TiDB 4.0 that automatically read data from TiKV or TiFlash through cost-based query optimization
Hardware configurations:
Component | Instances | vCPUs | RAM (GB) | Instance type |
TiKV and TiFlash | 3 | 16 | 122 | AWS i3.4xlarge |
TiDB | 2 | 16 | 32 | AWS c5.4xlarge |
Placement Driver (PD) | 3 | 4 | 16 | AWS m5.xlarge |
For configuration parameters and test steps, see this document.
Here are the TPC-H test results:
The TPC-H test results presented in a table are as follows. From the statistics, we can see that nearly all test queries were faster in TiDB 4.0.
Query | TiDB v3.0.13 (seconds) | v4.0 TiKV only (seconds) | v4.0 TiKV/TiFlash (seconds) | v4.0 TiKV only vs. v3.0.13 | v4.0 TiKV/TiFlash vs. v3.0.13 | ||
Speedup (%) | Speedup (times) | Speedup (%) | Speedup (times) | ||||
Q1 | 7.78 | 7.45 | 2.09 | 4.24 | 1.0 | 73.14 | 3.7 |
Q2 | 3.15 | 1.71 | 1.71 | 45.71 | 1.8 | 45.71 | 1.8 |
Q3 | 6.61 | 4.10 | 4.05 | 37.97 | 1.6 | 38.73 | 1.6 |
Q4 | 2.98 | 2.56 | 1.87 | 14.09 | 1.2 | 37.25 | 1.6 |
Q5 | 20.35 | 5.71 | 8.53 | 71.94 | 3.6 | 58.08 | 2.4 |
Q6 | 4.75 | 2.44 | 0.39 | 48.63 | 1.9 | 91.79 | 12.2 |
Q7 | 7.97 | 3.72 | 3.59 | 53.32 | 2.1 | 54.96 | 2.2 |
Q8 | 5.89 | 3.22 | 8.59 | 45.33 | 1.8 | -45.84 | 0.7 |
Q9 | 34.08 | 11.87 | 15.41 | 65.17 | 2.9 | 54.78 | 2.2 |
Q10 | 4.83 | 2.75 | 3.35 | 43.06 | 1.8 | 30.64 | 1.4 |
Q11 | 3.98 | 1.60 | 1.59 | 59.80 | 2.5 | 60.05 | 2.5 |
Q12 | 5.63 | 3.40 | 1.03 | 39.61 | 1.7 | 81.71 | 5.5 |
Q13 | 5.41 | 4.56 | 4.02 | 15.71 | 1.2 | 25.69 | 1.3 |
Q14 | 5.19 | 3.10 | 0.78 | 40.27 | 1.7 | 84.97 | 6.7 |
Q15 | 10.25 | 1.82 | 1.26 | 82.24 | 5.6 | 87.71 | 8.1 |
Q16 | 2.46 | 1.51 | 1.58 | 38.62 | 1.6 | 35.77 | 1.6 |
Q17 | 23.76 | 12.38 | 8.52 | 47.90 | 1.9 | 64.14 | 2.8 |
Q18 | 17.14 | 16.38 | 16.06 | 4.43 | 1.0 | 6.30 | 1.1 |
Q19 | 5.70 | 4.59 | 3.20 | 19.47 | 1.2 | 43.86 | 1.8 |
Q20 | 4.98 | 1.89 | 1.29 | 62.05 | 2.6 | 74.10 | 3.9 |
Q21 | 11.12 | 6.23 | 6.26 | 43.97 | 1.8 | 43.71 | 1.8 |
Q22 | 4.49 | 3.05 | 2.31 | 32.07 | 1.5 | 48.55 | 1.9 |
SUM | 198.50 | 106.04 | 97.48 | 46.58 | 1.9 | 50.89 | 2.0 |
TiDB 4.0’s query performance achieves noteworthy improvement because we introduce Chunk RPC to 4.0. Before TiKV sends data to TiDB, it organizes the data in columns. When TiDB receives the data, it can directly execute SQL statements without decoding. This avoids excessive encoding and decoding overheads when TiDB aggregates the data TiKV sends and thus significantly improves query execution performance.
TiDB’s Chunk structure
As our previous post described, at the end of 2017, we introduced the concept of a “Chunk.” A Chunk is composed of multiple columns.
There are two types of columns:
- Fixed-length columns, in which the data has a specified length that cannot be changed.
- Variable-length columns, in which the data length can change.
Regardless of the column type, data in columns are contiguously stored in memory in the Column.data
field (which is an array). If the data length varies, Column.offset
records the data offset. If the data is with fixed length, no offset is recorded.
The following figure illustrates the new vector access interface for Chunks:
- For fixed-length data, such as int64 numbers, the Golang
unsafe
package directly convertsColumn.data
to[]int64
inInt64s() []int64
, and returns the result. The user who wants to read or modifyColumn.data
can directly manipulate the array. This is the most efficient way to access fixed-length data. - For variable-length data, such as a string, we can use only
GetString(rowIdx int) string
to obtain the data in the corresponding row, and only append data to update it. Randomly modifying an element in the variable-length data column involves moving all the subsequent data. This creates a heavy overhead. To improve the overall performance, this operation is not implemented inColumn
.
Why Chunk RPC?
In version 3.0 and earlier, TiDB reads data from TiKV in the following default encoding format:
This encoding format is simple, but unfriendly to decoding:
- Under the existing execution framework, decoding each record requires multiple function calls. When the data size is large, the overhead of function calls is high.
- Decoding some types of data requires a lot of calculations. For example, decoding variable integers is more complicated than decoding integers. When decoding decimals, we need to calculate a lot of information like precision to restore the entire structure. This consumes more CPU resources while memory usage and network transmissions are only slightly reduced.
- In the decoding process, we need to construct a large number of objects and allocate a large amount of memory resources.
After TiDB received the data TiKV sent, it performed a large number of decoding operations to convert the data from the row format to the columnar Chunk format. Then, the data could be used for calculation. In OLAP scenarios where a lot of complex types of data such as Decimal and JSON were read, CPU overhead was high. In this case, data decoding might become a bottleneck for an SQL statement execution.
For example, here is a flame graph:
We found that the CPU runtime during data decoding accounted for 50% of the total CPU runtime. If we could reduce or even eliminate this CPU consumption, we could improve the SQL statement execution efficiency.
Therefore, we decided to introduce a more efficient encoding and decoding format for the Chunk structure. The use of this encoding format for data transmission is called Chunk RPC.
Chunk RPC in TiDB 4.0
TiDB 4.0 uses the following encoding format:
This encoding format is similar to TiDB Chunk’s calculation structure. This format eliminates the overhead of converting from row data to column data. This format is also friendly to decoding:
- A function call can decode a column of data. This greatly reduces the function call overhead.
- When TiKV encodes data of DECIMAL, TIME, JSON, and other types, it retains its structure. When TiDB decodes data, it obtains complete data objects without having to do extra calculations.
- Because this format is similar to Chunk, we can point the pointer inside Go Slice to
byte[]
, which is for decoding. Then, we can reuse the memory. This saves memory space and significantly reduces decoding overhead.
Chunk RPC does not impact OLTP capabilities
Note that Chunk RPC does not affect TiDB’s performance in OLTP scenarios.
We ran Sysbench and TPC-C benchmarks against the default decoding format and Chunk RPC to compare their throughput. The results showed that their queries per second (QPS) were almost the same:
Therefore, using the Chunk encoding format for data transmission does not impact TiDB’s OLTP capabilities.
Conclusion
In this post, we discussed how TiDB improves its transaction processing and query analytics capabilities from 3.0 to 4.0. If you’re interested, you can join our community on Slack and send us your feedback.
References:
[1] In Search of an Understandable Consensus Algorithm
[2] X-Engine: An Optimized Storage Engine for Large-scale E-commerce Transaction Processing
Experience modern data infrastructure firsthand.
TiDB Dedicated
A fully-managed cloud DBaaS for predictable workloads
TiDB Serverless
A fully-managed cloud DBaaS for auto-scaling workloads