Motivation
At present, the underlying storage in Doris is column storage. But query
execution needs to be transferred to the query layer for execution by
row-to-column first. Such an implementation maybe cause the performance problem。
*
* Column-to-row loss.
*
* Can not get better CPU performance without vectorized execution.
Currently, vectorized execution has been commonly adopted in mainstream MPP
databases, which has a significant effect on improving CPU utilization. In this
paper, we investigate the knowledge of vectorization and give a detailed design
for implementing vectorization on columnar storage.
What is vectorization execution
Organizing data in this batched, columnar fashion is the primary prerequisite
for using SIMD CPU instructions, which operate on a vector of data at a time.
Using SIMD instructions is an eventual goal.
Now suppose there is a table People, the data content is as follows:
Id Name Age
101 Ivan 22
115 Peggy 37
114 Victor 45
113 Eve 25
112 Walter 19
109 Trudy 31
108 Bob 27
105 Zoe 29
104 Charlie 42
102 Alice 35
Execute the following query on the table:SELECT Id, Name, Age, (Age - 30) * 50
AS Bonus FROM People WHERE Age > 30
Volcano model
[image.png]<https://camo.githubusercontent.com/d958b2f88b643b0ea00fc59b591b712e7ceb091591fbe07c1588ddbe861fb447/68747470733a2f2f75706c6f61642d696d616765732e6a69616e7368752e696f2f75706c6f61645f696d616765732f383535323230312d363735653739333964343334616161622e706e673f696d6167654d6f6772322f6175746f2d6f7269656e742f7374726970253743696d61676556696577322f322f772f31323430>
For traditional relational databases, the way it returns data is on a per-row
basis. The problems are: one function call per line, which interrupts CPU flow
and is not conducive to branch prediction; high instruction and data cache
misses; unfriendly compiler, not conducive to loop expansion, and not conducive
to using CPU acceleration instructions such as SIMD.
Vectorization Execution Engine
[image.png]<https://camo.githubusercontent.com/a26e4e3439257099951845b7c7992af564051a252f4dae9fe580e819cebb5962/68747470733a2f2f75706c6f61642d696d616765732e6a69616e7368752e696f2f75706c6f61645f696d616765732f383535323230312d373433323964303434346236643266392e706e673f696d6167654d6f6772322f6175746f2d6f7269656e742f7374726970253743696d61676556696577322f322f772f31323430>
But for columnar databases, you can actually do it in the form of columns,
which can reduce the overall cache failure rate. This is equivalent to a
vectorized execution to achieve such an effect, which is a unique optimization
technique for columnar databases to reduce CPU consumption and improve the
overall CPU utilization.
It can bring us the following benefits:
*
* Reduced dummy function calls for branch prediction.
*
* Reduces interpretation overhead by getting a batch of results at a
time.
*
* Easy for compiler to do loop pipelining and SIMD optimization.
*
* Friendly to CPU L1 and L2 Cache.
How Vectorization Execution Engine For Doris
The key idea and challenge
* Design a new memory structure to replace the original RowBatch and Tuple
structure
* Rewrite all operators to support vectorization/columnar computation
1. Data Structure
We have implemented all types natively supported by Doris in Column MemLayout
* tinyint
* smallint
* int
* bigint
* largeint
* boolean
* float
* double
* decimal
* date
* datetime
* HLL
* bitmap
2. Function and AggregateFunction
We have implemented more than 80% of Doris's native support functions
implementation of Vectorized Function and AggregateFunction interfaces, and
have completed a large number of SIMD support.
And for a more efficient SIMD optimization, the type of NULLABLE for each
function is identified as follows.
[image.png]<https://camo.githubusercontent.com/69d48e92d7b88c3271084c8716e86031a43169789adf4d7a6c683cd2d45eb922/68747470733a2f2f75706c6f61642d696d616765732e6a69616e7368752e696f2f75706c6f61645f696d616765732f383535323230312d313533356338373536643134623434372e706e673f696d6167654d6f6772322f6175746f2d6f7269656e742f7374726970253743696d61676556696577322f322f772f31323430>
3. Operator
Operator
After the data is read from the storage layer, it is transformed from RowCursor
to Vectorized Column structures, i.e., from row storage to column storage.
Starting from VOlapScanNode, the data is organized in the form of columns.
[image.png]<https://camo.githubusercontent.com/5ae60da39b57f312f2595458f039172fba5766a2a9f5aa4d57d9379b52ab2ba3/68747470733a2f2f75706c6f61642d696d616765732e6a69616e7368752e696f2f75706c6f61645f696d616765732f383535323230312d643930636662666361616533323432362e706e673f696d6167654d6f6772322f6175746f2d6f7269656e742f7374726970253743696d61676556696577322f322f772f31323430>
4. Interface
Later we will gradually implement some basic vectorized interfaces and submit
them to the community.
Thanks,
Li HaoPeng