Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

executor/join : use shallow copy for join. #7433

Merged
merged 49 commits into from
Aug 29, 2018
Merged
Show file tree
Hide file tree
Changes from 37 commits
Commits
Show all changes
49 commits
Select commit Hold shift + click to select a range
a576cc0
iterator copy init
crazycs520 Aug 19, 2018
79b74e7
fix iterator bug
crazycs520 Aug 19, 2018
d9fe98b
fix nullmap index out of range and add test
crazycs520 Aug 19, 2018
8fa2ad8
refine code
crazycs520 Aug 19, 2018
33ae5a0
refine code
crazycs520 Aug 19, 2018
255c64b
remove iterator copy and use back to pre rows
crazycs520 Aug 19, 2018
0973373
checkout joiner.go file
crazycs520 Aug 19, 2018
055a41a
add check to bench
crazycs520 Aug 20, 2018
c5eeff9
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 20, 2018
e7c6c64
iterator only once
crazycs520 Aug 21, 2018
ce8eb4f
add appendMultiSameNullBitmap
crazycs520 Aug 21, 2018
499b6f9
field by field only one line 2X
crazycs520 Aug 22, 2018
2a295a1
refine column copy
crazycs520 Aug 22, 2018
9d82447
refine column copy
crazycs520 Aug 22, 2018
ef85948
add shadow copy to join and move code
crazycs520 Aug 22, 2018
45b4631
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 22, 2018
5bf279f
rename function
crazycs520 Aug 22, 2018
9690506
add comment
crazycs520 Aug 22, 2018
8db639f
add shadow copy to inner join
crazycs520 Aug 22, 2018
7a55ff5
refine code
crazycs520 Aug 22, 2018
05c1273
add shadow copy to all join
crazycs520 Aug 22, 2018
66a133c
remove redundancy code
crazycs520 Aug 22, 2018
dadb047
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 22, 2018
b4192e4
remove column copy and redundancy code
crazycs520 Aug 23, 2018
4096997
address comment
crazycs520 Aug 23, 2018
b802941
add mutchunk
crazycs520 Aug 23, 2018
c5cfdf1
address comment
crazycs520 Aug 23, 2018
947f9d4
use mutRow instead of mut chunk.
crazycs520 Aug 23, 2018
24ab90e
address comment
crazycs520 Aug 23, 2018
2b8d896
refine code
crazycs520 Aug 23, 2018
3f82d2b
address comment
crazycs520 Aug 23, 2018
604e49d
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 23, 2018
e5f4cbe
address comment
crazycs520 Aug 23, 2018
abbc2c9
address comment
crazycs520 Aug 24, 2018
e1dd31d
address comment and add test to mutRow_test
crazycs520 Aug 24, 2018
593b31c
remove chunk_copy_test.go
crazycs520 Aug 24, 2018
3a6fbb7
refine code
crazycs520 Aug 24, 2018
600fdc3
refine test
crazycs520 Aug 24, 2018
f4fbd70
refine test and code
crazycs520 Aug 24, 2018
23eaf1e
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 24, 2018
e9ef7dd
optimize append num
crazycs520 Aug 24, 2018
21b5417
remove shadown copy on inner, leftOut, rightOut join, vectorized filt…
crazycs520 Aug 27, 2018
0aadbf6
address comment
crazycs520 Aug 27, 2018
0de2063
address comment
crazycs520 Aug 27, 2018
c681658
address comment
crazycs520 Aug 28, 2018
f8ccdf2
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 28, 2018
c7b2301
Merge branch 'master' of https://github.com/pingcap/tidb into column-…
crazycs520 Aug 29, 2018
1e8a9f0
update test after merge
crazycs520 Aug 29, 2018
b939b3b
Merge branch 'master' into column-copy
XuHuaiyu Aug 29, 2018
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
120 changes: 43 additions & 77 deletions executor/joiner.go
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ func newJoiner(ctx sessionctx.Context, joinType plan.JoinType,
colTypes := make([]*types.FieldType, 0, len(lhsColTypes)+len(rhsColTypes))
colTypes = append(colTypes, lhsColTypes...)
colTypes = append(colTypes, rhsColTypes...)
base.chk = chunk.NewChunkWithCapacity(colTypes, ctx.GetSessionVars().MaxChunkSize)
base.mutRow = chunk.MutRowFromTypes(colTypes)
base.selected = make([]bool, 0, chunk.InitialCapacity)
if joinType == plan.LeftOuterJoin || joinType == plan.RightOuterJoin {
innerColTypes := lhsColTypes
Expand Down Expand Up @@ -124,7 +124,7 @@ type baseJoiner struct {
conditions []expression.Expression
defaultInner chunk.Row
outerIsRight bool
chk *chunk.Chunk
mutRow chunk.MutRow
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

how about s/mutRow/shadowRow/?

selected []bool
maxChunkSize int
}
Expand All @@ -142,6 +142,16 @@ func (j *baseJoiner) makeJoinRowToChunk(chk *chunk.Chunk, lhs, rhs chunk.Row) {
chk.AppendPartialRow(lhs.Len(), rhs)
}

// makeJoinRow combines inner, outer row into mutRow.
// combines will uses shadow copy inner and outer row data to mutRow.
func (j *baseJoiner) makeJoinRow(isRightJoin bool, inner, outer chunk.Row) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

how about s/makeJoinRow/makeShallowJoinRow/

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done.

if !isRightJoin {
inner, outer = outer, inner
}
j.mutRow.ShadowCopyPartialRow(0, inner)
j.mutRow.ShadowCopyPartialRow(inner.Len(), outer)
}

func (j *baseJoiner) filter(input, output *chunk.Chunk) (matched bool, err error) {
j.selected, err = expression.VectorizedFilter(j.ctx, j.conditions, chunk.NewIterator4Chunk(input), j.selected)
if err != nil {
Expand Down Expand Up @@ -173,14 +183,9 @@ func (j *semiJoiner) tryToMatch(outer chunk.Row, inners chunk.Iterator, chk *chu
}

for inner := inners.Current(); inner != inners.End(); inner = inners.Next() {
j.chk.Reset()
if j.outerIsRight {
j.makeJoinRowToChunk(j.chk, inner, outer)
} else {
j.makeJoinRowToChunk(j.chk, outer, inner)
}
j.makeJoinRow(j.outerIsRight, inner, outer)

matched, err = expression.EvalBool(j.ctx, j.conditions, j.chk.GetRow(0))
matched, err = expression.EvalBool(j.ctx, j.conditions, j.mutRow.ToRow())
if err != nil {
return false, errors.Trace(err)
}
Expand Down Expand Up @@ -212,14 +217,9 @@ func (j *antiSemiJoiner) tryToMatch(outer chunk.Row, inners chunk.Iterator, chk
}

for inner := inners.Current(); inner != inners.End(); inner = inners.Next() {
j.chk.Reset()
if j.outerIsRight {
j.makeJoinRowToChunk(j.chk, inner, outer)
} else {
j.makeJoinRowToChunk(j.chk, outer, inner)
}
j.makeJoinRow(j.outerIsRight, inner, outer)

matched, err = expression.EvalBool(j.ctx, j.conditions, j.chk.GetRow(0))
matched, err = expression.EvalBool(j.ctx, j.conditions, j.mutRow.ToRow())
if err != nil {
return false, errors.Trace(err)
}
Expand Down Expand Up @@ -252,10 +252,9 @@ func (j *leftOuterSemiJoiner) tryToMatch(outer chunk.Row, inners chunk.Iterator,
}

for inner := inners.Current(); inner != inners.End(); inner = inners.Next() {
j.chk.Reset()
j.makeJoinRowToChunk(j.chk, outer, inner)
j.makeJoinRow(false, inner, outer)

matched, err = expression.EvalBool(j.ctx, j.conditions, j.chk.GetRow(0))
matched, err = expression.EvalBool(j.ctx, j.conditions, j.mutRow.ToRow())
if err != nil {
return false, errors.Trace(err)
}
Expand Down Expand Up @@ -295,10 +294,9 @@ func (j *antiLeftOuterSemiJoiner) tryToMatch(outer chunk.Row, inners chunk.Itera
}

for inner := inners.Current(); inner != inners.End(); inner = inners.Next() {
j.chk.Reset()
j.makeJoinRowToChunk(j.chk, outer, inner)
matched, err := expression.EvalBool(j.ctx, j.conditions, j.chk.GetRow(0))
j.makeJoinRow(false, inner, outer)

matched, err := expression.EvalBool(j.ctx, j.conditions, j.mutRow.ToRow())
if err != nil {
return false, errors.Trace(err)
}
Expand Down Expand Up @@ -330,25 +328,7 @@ func (j *leftOuterJoiner) tryToMatch(outer chunk.Row, inners chunk.Iterator, chk
if inners.Len() == 0 {
return false, nil
}

j.chk.Reset()
chkForJoin := j.chk
if len(j.conditions) == 0 {
chkForJoin = chk
}

numToAppend := j.maxChunkSize - chk.NumRows()
for ; inners.Current() != inners.End() && numToAppend > 0; numToAppend-- {
j.makeJoinRowToChunk(chkForJoin, outer, inners.Current())
inners.Next()
}
if len(j.conditions) == 0 {
return true, nil
}

// reach here, chkForJoin is j.chk
matched, err := j.filter(chkForJoin, chk)
return matched, errors.Trace(err)
return j.tryToMatchInnerAndOuter(false, outer, inners, chk)
}

func (j *leftOuterJoiner) onMissMatch(outer chunk.Row, chk *chunk.Chunk) {
Expand All @@ -366,24 +346,7 @@ func (j *rightOuterJoiner) tryToMatch(outer chunk.Row, inners chunk.Iterator, ch
return false, nil
}

j.chk.Reset()
chkForJoin := j.chk
if len(j.conditions) == 0 {
chkForJoin = chk
}

numToAppend := j.maxChunkSize - chk.NumRows()
for ; inners.Current() != inners.End() && numToAppend > 0; numToAppend-- {
j.makeJoinRowToChunk(chkForJoin, inners.Current(), outer)
inners.Next()
}
if len(j.conditions) == 0 {
return true, nil
}

// reach here, chkForJoin is j.chk
matched, err := j.filter(chkForJoin, chk)
return matched, errors.Trace(err)
return j.tryToMatchInnerAndOuter(true, outer, inners, chk)
}

func (j *rightOuterJoiner) onMissMatch(outer chunk.Row, chk *chunk.Chunk) {
Expand All @@ -400,26 +363,29 @@ func (j *innerJoiner) tryToMatch(outer chunk.Row, inners chunk.Iterator, chk *ch
if inners.Len() == 0 {
return false, nil
}
j.chk.Reset()
chkForJoin := j.chk
if len(j.conditions) == 0 {
chkForJoin = chk
}
inner, numToAppend := inners.Current(), j.maxChunkSize-chk.NumRows()
for ; inner != inners.End() && numToAppend > 0; inner, numToAppend = inners.Next(), numToAppend-1 {
if j.outerIsRight {
j.makeJoinRowToChunk(chkForJoin, inner, outer)
} else {
j.makeJoinRowToChunk(chkForJoin, outer, inner)

return j.tryToMatchInnerAndOuter(j.outerIsRight, outer, inners, chk)
}

// tryToMatchInnerAndOuter does 2 things:
// 1. Combine outer and inner row to join row.
// 2. Check whether the join row matches the join conditions, if so, append it to the `outChk`.
func (j *baseJoiner) tryToMatchInnerAndOuter(isRight bool, outer chunk.Row, inners chunk.Iterator, outChk *chunk.Chunk) (bool, error) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the execution logic of this function can be split to 2 stages:

  1. the first stage joins each inner row with the only outer row, computes the filter result of the joined row, stores the result matched into a bool slice. With this bool slice we can know whether an inner row can be joined with the only outer row.
  2. the second stage does the real deep copy work: according to the bool slice from stage 1, deep-copy the outer and matched inner row into the destination Chunk column by column.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe we also can use this pattern in selectExec and other match then output execs as well if it works.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Exactly, Nice catch!

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe we should use an individual issue to trace and record the enhancement of the cache locality in the execution engine. But it's not easy to find the potential optimization opportunities without reading and analyzing the code. It may take us a long time to finish it.

match := false
numToAppend := j.maxChunkSize - outChk.NumRows()
for inner := inners.Current(); inner != inners.End() && numToAppend > 0; inner, numToAppend = inners.Next(), numToAppend-1 {
j.makeJoinRow(isRight, inner, outer)

matched, err := expression.EvalBool(j.ctx, j.conditions, j.mutRow.ToRow())
if err != nil {
return false, errors.Trace(err)
}
if matched {
match = true
outChk.AppendRow(j.mutRow.ToRow())
}
}
if len(j.conditions) == 0 {
return true, nil
}

// reach here, chkForJoin is j.chk
matched, err := j.filter(chkForJoin, chk)
return matched, errors.Trace(err)
return match, nil
}

func (j *innerJoiner) onMissMatch(outer chunk.Row, chk *chunk.Chunk) {
Expand Down
2 changes: 1 addition & 1 deletion util/chunk/chunk.go
Original file line number Diff line number Diff line change
Expand Up @@ -182,7 +182,7 @@ func (c *Chunk) Append(other *Chunk, begin, end int) {
c.numVirtualRows += end - begin
}

// TruncateTo truncates rows from tail to head in a Chunk to "numRows" rows.
// TruncateTo truncates rows from tail to head in a Chunk to "rowsNum" rows.
func (c *Chunk) TruncateTo(numRows int) {
for _, col := range c.columns {
if col.isFixed() {
Expand Down
12 changes: 12 additions & 0 deletions util/chunk/chunk_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -256,6 +256,18 @@ func newChunk(elemLen ...int) *Chunk {
return chk
}

func newChunkWithInitCap(cap int, elemLen ...int) *Chunk {
chk := &Chunk{}
for _, l := range elemLen {
if l > 0 {
chk.addFixedLenColumn(l, cap)
} else {
chk.addVarLenColumn(cap)
}
}
return chk
}

var allTypes = []*types.FieldType{
types.NewFieldType(mysql.TypeTiny),
types.NewFieldType(mysql.TypeShort),
Expand Down
23 changes: 23 additions & 0 deletions util/chunk/mutrow.go
Original file line number Diff line number Diff line change
Expand Up @@ -346,3 +346,26 @@ func setMutRowJSON(col *column, j json.BinaryJSON) {
copy(col.data[1:], j.Value)
col.offsets[1] = int32(dataLen)
}

// ShadowCopyPartialRow shadow copies the data of `row` to MutRow.
func (mr MutRow) ShadowCopyPartialRow(colIdx int, row Row) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/ShadowCopy/ShallowCopy/

chk := mr.c
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

chk is only used once, I think we can remove this variable and use mr.c directly.

for i, rowCol := range row.c.columns {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

how about:
s/rowCol/srcCol/
s/chkCol/dstCol/

chkCol := chk.columns[colIdx+i]
if !rowCol.isNull(row.idx) {
chkCol.nullBitmap[0] = 1
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's better to add some comments about why we set the whole byte to 1/0

} else {
chkCol.nullBitmap[0] = 0
}

if rowCol.isFixed() {
elemLen := len(rowCol.elemBuf)
offset := row.idx * elemLen
chkCol.data = rowCol.data[offset : offset+elemLen]
} else {
start, end := rowCol.offsets[row.idx], rowCol.offsets[row.idx+1]
chkCol.data = rowCol.data[start:end]
chkCol.offsets[1] = int32(len(chkCol.data))
}
}
}
50 changes: 50 additions & 0 deletions util/chunk/mutrow_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -134,3 +134,53 @@ func BenchmarkMutRowFromValues(b *testing.B) {
MutRowFromValues(values)
}
}

func (s *testChunkSuite) TestMutRowShadowCopyPartialRow(c *check.C) {
colTypes := make([]*types.FieldType, 0, 8)
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeVarString})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeVarString})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeLonglong})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeLonglong})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeDatetime})

mutRow := MutRowFromTypes(colTypes)
row := MutRowFromValues("abc", "abcdefg", 123, 456, types.ZeroDatetime).ToRow()
mutRow.ShadowCopyPartialRow(0, row)
c.Assert(row.GetString(0), check.DeepEquals, mutRow.ToRow().GetString(0))
c.Assert(row.GetString(1), check.DeepEquals, mutRow.ToRow().GetString(1))
c.Assert(row.GetInt64(2), check.DeepEquals, mutRow.ToRow().GetInt64(2))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why we use DeepEquals here?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We'd better modify a value of row, and then check whether the corresponding value in mutRow is also modified.

c.Assert(row.GetInt64(3), check.DeepEquals, mutRow.ToRow().GetInt64(3))
c.Assert(row.GetTime(4), check.DeepEquals, mutRow.ToRow().GetTime(4))
}

var rowsNum = 1024

func BenchmarkMutRowShadowCopyPartialRow(b *testing.B) {
b.ReportAllocs()
colTypes := make([]*types.FieldType, 0, 8)
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeVarString})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeVarString})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeLonglong})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeLonglong})
colTypes = append(colTypes, &types.FieldType{Tp: mysql.TypeDatetime})

mutRow := MutRowFromTypes(colTypes)
row := MutRowFromValues("abc", "abcdefg", 123, 456, types.ZeroDatetime).ToRow()
for i := 0; i < b.N; i++ {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

b.ResetTimer() before this loop.

for j := 0; j < rowsNum; j++ {
mutRow.ShadowCopyPartialRow(0, row)
}
}
}

func BenchmarkChunkAppendPartialRow(b *testing.B) {
b.ReportAllocs()
chk := newChunkWithInitCap(rowsNum, 0, 0, 8, 8, 16)
row := MutRowFromValues("abc", "abcdefg", 123, 456, types.ZeroDatetime).ToRow()
for i := 0; i < b.N; i++ {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ditto

chk.Reset()
for j := 0; j < rowsNum; j++ {
chk.AppendPartialRow(0, row)
}
}
}