90 Star 494 Fork 152

平凯星辰(北京)科技有限公司/tidb

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
physical_plan_builder.go 19.19 KB
一键复制 编辑 原始数据 按行查看 历史
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619
// Copyright 2017 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// See the License for the specific language governing permissions and
// limitations under the License.
package plan
import (
"math"
"github.com/juju/errors"
"github.com/pingcap/tidb/expression"
"github.com/pingcap/tidb/infoschema"
"github.com/pingcap/tidb/model"
"github.com/pingcap/tidb/mysql"
"github.com/pingcap/tidb/types"
"github.com/pingcap/tidb/util/ranger"
log "github.com/sirupsen/logrus"
)
const (
netWorkFactor = 1.5
netWorkStartFactor = 20.0
scanFactor = 2.0
descScanFactor = 5 * scanFactor
memoryFactor = 5.0
hashAggMemFactor = 2.0
selectionFactor = 0.8
distinctFactor = 0.8
cpuFactor = 0.9
)
// wholeTaskTypes records all possible kinds of task that a plan can return. For Agg, TopN and Limit, we will try to get
// these tasks one by one.
var wholeTaskTypes = [...]taskType{copSingleReadTaskType, copDoubleReadTaskType, rootTaskType}
var invalidTask = &rootTask{cst: math.MaxFloat64}
// getPropByOrderByItems will check if this sort property can be pushed or not. In order to simplify the problem, we only
// consider the case that all expression are columns and all of them are asc or desc.
func getPropByOrderByItems(items []*ByItems) (*requiredProp, bool) {
desc := false
cols := make([]*expression.Column, 0, len(items))
for i, item := range items {
col, ok := item.Expr.(*expression.Column)
if !ok {
return nil, false
}
cols = append(cols, col)
desc = item.Desc
if i > 0 && item.Desc != items[i-1].Desc {
return nil, false
}
}
return &requiredProp{cols: cols, desc: desc}, true
}
func (p *LogicalTableDual) findBestTask(prop *requiredProp) (task, error) {
if !prop.isEmpty() {
return invalidTask, nil
}
dual := PhysicalTableDual{RowCount: p.RowCount}.init(p.ctx, p.stats)
dual.SetSchema(p.schema)
return &rootTask{p: dual}, nil
}
// findBestTask implements LogicalPlan interface.
func (p *baseLogicalPlan) findBestTask(prop *requiredProp) (bestTask task, err error) {
// Look up the task with this prop in the task map.
// It's used to reduce double counting.
bestTask = p.getTask(prop)
if bestTask != nil {
return bestTask, nil
}
if prop.taskTp != rootTaskType {
// Currently all plan cannot totally push down.
p.storeTask(prop, invalidTask)
return invalidTask, nil
}
bestTask = invalidTask
childTasks := make([]task, 0, len(p.children))
for _, pp := range p.self.exhaustPhysicalPlans(prop) {
// find best child tasks firstly.
childTasks = childTasks[:0]
for i, child := range p.children {
childTask, err := child.findBestTask(pp.getChildReqProps(i))
if err != nil {
return nil, errors.Trace(err)
}
childTasks = append(childTasks, childTask)
}
// combine best child tasks with parent physical plan.
curTask := pp.attach2Task(childTasks...)
// get the most efficient one.
if curTask.cost() < bestTask.cost() {
bestTask = curTask
}
}
p.storeTask(prop, bestTask)
return bestTask, nil
}
// tryToGetMemTask will check if this table is a mem table. If it is, it will produce a task.
func (ds *DataSource) tryToGetMemTask(prop *requiredProp) (task task, err error) {
if !prop.isEmpty() {
return nil, nil
}
if !infoschema.IsMemoryDB(ds.DBName.L) {
return nil, nil
}
memTable := PhysicalMemTable{
DBName: ds.DBName,
Table: ds.tableInfo,
Columns: ds.Columns,
TableAsName: ds.TableAsName,
}.init(ds.ctx, ds.stats)
memTable.SetSchema(ds.schema)
memTable.Ranges = ranger.FullIntRange()
// Stop to push down these conditions.
var retPlan PhysicalPlan = memTable
if len(ds.pushedDownConds) > 0 {
sel := PhysicalSelection{
Conditions: ds.pushedDownConds,
}.init(ds.ctx, ds.stats)
sel.SetChildren(memTable)
retPlan = sel
}
return &rootTask{p: retPlan}, nil
}
// tryToGetDualTask will check if the push down predicate has false constant. If so, it will return table dual.
func (ds *DataSource) tryToGetDualTask() (task, error) {
for _, cond := range ds.pushedDownConds {
if _, ok := cond.(*expression.Constant); ok {
result, err := expression.EvalBool(ds.ctx, []expression.Expression{cond}, nil)
if err != nil {
return nil, errors.Trace(err)
}
if !result {
dual := PhysicalTableDual{}.init(ds.ctx, ds.stats)
dual.SetSchema(ds.schema)
return &rootTask{
p: dual,
}, nil
}
}
}
return nil, nil
}
// findBestTask implements the PhysicalPlan interface.
// It will enumerate all the available indices and choose a plan with least cost.
func (ds *DataSource) findBestTask(prop *requiredProp) (task, error) {
// If ds is an inner plan in an IndexJoin, the IndexJoin will generate an inner plan by itself.
// So here we do nothing.
// TODO: Add a special prop to handle IndexJoin's inner plan.
// Then we can remove forceToTableScan and forceToIndexScan.
if prop == nil {
return nil, nil
}
t := ds.getTask(prop)
if t != nil {
return t, nil
}
t, err := ds.tryToGetDualTask()
if err != nil {
return nil, errors.Trace(err)
}
if t != nil {
ds.storeTask(prop, t)
return t, nil
}
t, err = ds.tryToGetMemTask(prop)
if err != nil {
return nil, errors.Trace(err)
}
if t != nil {
ds.storeTask(prop, t)
return t, nil
}
indices := ds.availableIndices.indices
includeTableScan := ds.availableIndices.includeTableScan
t = invalidTask
if includeTableScan {
t, err = ds.convertToTableScan(prop)
if err != nil {
return nil, errors.Trace(err)
}
}
if !includeTableScan || len(ds.pushedDownConds) > 0 || len(prop.cols) > 0 {
for i, idx := range indices {
// TODO: We can also check if the prop matches the index columns.
if !ds.relevantIndices[i] && len(prop.cols) == 0 {
continue
}
idxTask, err := ds.convertToIndexScan(prop, idx)
if err != nil {
return nil, errors.Trace(err)
}
if idxTask.cost() < t.cost() {
t = idxTask
}
}
}
ds.storeTask(prop, t)
return t, nil
}
func isCoveringIndex(columns []*model.ColumnInfo, indexColumns []*model.IndexColumn, pkIsHandle bool) bool {
for _, colInfo := range columns {
if pkIsHandle && mysql.HasPriKeyFlag(colInfo.Flag) {
continue
}
if colInfo.ID == model.ExtraHandleID {
continue
}
isIndexColumn := false
for _, indexCol := range indexColumns {
isFullLen := indexCol.Length == types.UnspecifiedLength || indexCol.Length == colInfo.Flen
if colInfo.Name.L == indexCol.Name.L && isFullLen {
isIndexColumn = true
break
}
}
if !isIndexColumn {
return false
}
}
return true
}
func (ds *DataSource) forceToIndexScan(idx *model.IndexInfo, remainedConds []expression.Expression) PhysicalPlan {
is := PhysicalIndexScan{
Table: ds.tableInfo,
TableAsName: ds.TableAsName,
DBName: ds.DBName,
Columns: ds.Columns,
Index: idx,
dataSourceSchema: ds.schema,
Ranges: ranger.FullNewRange(),
KeepOrder: false,
}.init(ds.ctx)
is.filterCondition = remainedConds
is.stats = ds.stats
cop := &copTask{
indexPlan: is,
}
if !isCoveringIndex(is.Columns, is.Index.Columns, is.Table.PKIsHandle) {
// On this way, it's double read case.
ts := PhysicalTableScan{Columns: ds.Columns, Table: is.Table}.init(ds.ctx)
ts.SetSchema(is.dataSourceSchema)
cop.tablePlan = ts
}
is.initSchema(ds.id, idx, cop.tablePlan != nil)
is.addPushedDownSelection(cop, ds, math.MaxFloat64)
t := finishCopTask(ds.ctx, cop)
return t.plan()
}
// If there is a table reader which needs to keep order, we should append a pk to table scan.
func (ts *PhysicalTableScan) appendExtraHandleCol(ds *DataSource) {
if len(ds.schema.TblID2Handle) > 0 {
return
}
pkInfo := model.NewExtraHandleColInfo()
ts.Columns = append(ts.Columns, pkInfo)
handleCol := ds.newExtraHandleSchemaCol()
ts.schema.Append(handleCol)
ts.schema.TblID2Handle[ds.tableInfo.ID] = []*expression.Column{handleCol}
}
// convertToIndexScan converts the DataSource to index scan with idx.
func (ds *DataSource) convertToIndexScan(prop *requiredProp, idx *model.IndexInfo) (task task, err error) {
is := PhysicalIndexScan{
Table: ds.tableInfo,
TableAsName: ds.TableAsName,
DBName: ds.DBName,
Columns: ds.Columns,
Index: idx,
dataSourceSchema: ds.schema,
}.init(ds.ctx)
statsTbl := ds.statisticTable
if statsTbl.Indices[idx.ID] != nil {
is.Hist = &statsTbl.Indices[idx.ID].Histogram
}
rowCount := float64(statsTbl.Count)
sc := ds.ctx.GetSessionVars().StmtCtx
idxCols, colLengths := expression.IndexInfo2Cols(ds.Schema().Columns, idx)
is.Ranges = ranger.FullNewRange()
eqCount := 0
if len(ds.pushedDownConds) > 0 {
is.conditions = ds.pushedDownConds
if len(idxCols) > 0 {
is.Ranges, is.AccessCondition, is.filterCondition, eqCount, err = ranger.DetachCondAndBuildRangeForIndex(ds.ctx, ds.pushedDownConds, idxCols, colLengths)
if err != nil {
return nil, errors.Trace(err)
}
rowCount, err = statsTbl.GetRowCountByIndexRanges(sc, is.Index.ID, is.Ranges)
if err != nil {
return nil, errors.Trace(err)
}
} else {
is.filterCondition = ds.pushedDownConds
}
}
cop := &copTask{indexPlan: is}
if !isCoveringIndex(is.Columns, is.Index.Columns, is.Table.PKIsHandle) {
// On this way, it's double read case.
ts := PhysicalTableScan{Columns: ds.Columns, Table: is.Table}.init(ds.ctx)
ts.SetSchema(ds.schema.Clone())
cop.tablePlan = ts
// If it's parent requires single read task, return max cost.
if prop.taskTp == copSingleReadTaskType {
return &copTask{cst: math.MaxFloat64}, nil
}
} else if prop.taskTp == copDoubleReadTaskType {
// If it's parent requires double read task, return max cost.
return &copTask{cst: math.MaxFloat64}, nil
}
is.initSchema(ds.id, idx, cop.tablePlan != nil)
// Check if this plan matches the property.
matchProperty := false
if !prop.isEmpty() {
for i, col := range idx.Columns {
// not matched
if col.Name.L == prop.cols[0].ColName.L {
matchProperty = matchIndicesProp(idx.Columns[i:], prop.cols)
break
} else if i >= eqCount {
break
}
}
}
// Only use expectedCnt when it's smaller than the count we calculated.
// e.g. IndexScan(count1)->After Filter(count2). The `ds.statsAfterSelect.count` is count2. count1 is the one we need to calculate
// If expectedCnt and count2 are both zero and we go into the below `if` block, the count1 will be set to zero though it's shouldn't be.
if matchProperty && prop.expectedCnt < ds.statsAfterSelect.count {
selectivity, err := statsTbl.Selectivity(ds.ctx, is.filterCondition)
if err != nil {
log.Warnf("An error happened: %v, we have to use the default selectivity", err.Error())
selectivity = selectionFactor
}
rowCount = math.Min(prop.expectedCnt/selectivity, rowCount)
}
is.stats = ds.stats.scaleByExpectCnt(rowCount)
cop.cst = rowCount * scanFactor
task = cop
if matchProperty {
if prop.desc {
is.Desc = true
cop.cst = rowCount * descScanFactor
}
if cop.tablePlan != nil {
cop.tablePlan.(*PhysicalTableScan).appendExtraHandleCol(ds)
}
cop.keepOrder = true
is.KeepOrder = true
is.addPushedDownSelection(cop, ds, prop.expectedCnt)
} else {
expectedCnt := math.MaxFloat64
if prop.isEmpty() {
expectedCnt = prop.expectedCnt
} else {
return invalidTask, nil
}
is.addPushedDownSelection(cop, ds, expectedCnt)
}
if prop.taskTp == rootTaskType {
task = finishCopTask(ds.ctx, task)
} else if _, ok := task.(*rootTask); ok {
return invalidTask, nil
}
return task, nil
}
func (is *PhysicalIndexScan) initSchema(id int, idx *model.IndexInfo, isDoubleRead bool) {
indexCols := make([]*expression.Column, 0, len(idx.Columns))
for _, col := range idx.Columns {
indexCols = append(indexCols, &expression.Column{FromID: id, Position: col.Offset})
}
setHandle := false
for _, col := range is.Columns {
if (mysql.HasPriKeyFlag(col.Flag) && is.Table.PKIsHandle) || col.ID == model.ExtraHandleID {
indexCols = append(indexCols, &expression.Column{FromID: id, ID: col.ID, Position: col.Offset})
setHandle = true
break
}
}
// If it's double read case, the first index must return handle. So we should add extra handle column
// if there isn't a handle column.
if isDoubleRead && !setHandle {
indexCols = append(indexCols, &expression.Column{FromID: id, ID: model.ExtraHandleID, Position: -1})
}
is.SetSchema(expression.NewSchema(indexCols...))
}
func (is *PhysicalIndexScan) addPushedDownSelection(copTask *copTask, p *DataSource, expectedCnt float64) {
// Add filter condition to table plan now.
if len(is.filterCondition) > 0 {
var indexConds, tableConds []expression.Expression
if copTask.tablePlan != nil {
indexConds, tableConds = splitIndexFilterConditions(is.filterCondition, is.Index.Columns, is.Table)
} else {
indexConds = is.filterCondition
}
if indexConds != nil {
indexSel := PhysicalSelection{Conditions: indexConds}.init(is.ctx,
p.getStatsByFilter(append(is.AccessCondition, indexConds...)).scaleByExpectCnt(expectedCnt))
indexSel.SetChildren(is)
copTask.indexPlan = indexSel
copTask.cst += copTask.count() * cpuFactor
}
if tableConds != nil {
copTask.finishIndexPlan()
tableSel := PhysicalSelection{Conditions: tableConds}.init(is.ctx, p.statsAfterSelect.scaleByExpectCnt(expectedCnt))
tableSel.SetChildren(copTask.tablePlan)
copTask.tablePlan = tableSel
copTask.cst += copTask.count() * cpuFactor
}
}
}
func matchIndicesProp(idxCols []*model.IndexColumn, propCols []*expression.Column) bool {
if len(idxCols) < len(propCols) {
return false
}
for i, col := range propCols {
if idxCols[i].Length != types.UnspecifiedLength || col.ColName.L != idxCols[i].Name.L {
return false
}
}
return true
}
func splitIndexFilterConditions(conditions []expression.Expression, indexColumns []*model.IndexColumn,
table *model.TableInfo) (indexConds, tableConds []expression.Expression) {
var pkName model.CIStr
if table.PKIsHandle {
pkInfo := table.GetPkColInfo()
if pkInfo != nil {
pkName = pkInfo.Name
}
}
var indexConditions, tableConditions []expression.Expression
for _, cond := range conditions {
if checkIndexCondition(cond, indexColumns, pkName) {
indexConditions = append(indexConditions, cond)
} else {
tableConditions = append(tableConditions, cond)
}
}
return indexConditions, tableConditions
}
// checkIndexCondition will check whether all columns of condition is index columns or primary key column.
func checkIndexCondition(condition expression.Expression, indexColumns []*model.IndexColumn, pkName model.CIStr) bool {
cols := expression.ExtractColumns(condition)
for _, col := range cols {
if pkName.L == col.ColName.L {
continue
}
isIndexColumn := false
for _, indCol := range indexColumns {
if col.ColName.L == indCol.Name.L && indCol.Length == types.UnspecifiedLength {
isIndexColumn = true
break
}
}
if !isIndexColumn {
return false
}
}
return true
}
func (ds *DataSource) forceToTableScan(pk *expression.Column) PhysicalPlan {
var ranges []*ranger.NewRange
if pk != nil {
ranges = ranger.FullIntNewRange(mysql.HasUnsignedFlag(pk.RetType.Flag))
} else {
ranges = ranger.FullIntNewRange(false)
}
ts := PhysicalTableScan{
Table: ds.tableInfo,
Columns: ds.Columns,
TableAsName: ds.TableAsName,
DBName: ds.DBName,
Ranges: ranges,
}.init(ds.ctx)
ts.SetSchema(ds.schema)
ts.stats = ds.stats
ts.filterCondition = ds.pushedDownConds
copTask := &copTask{
tablePlan: ts,
indexPlanFinished: true,
}
ts.addPushedDownSelection(copTask, ds.stats)
t := finishCopTask(ds.ctx, copTask)
return t.plan()
}
// convertToTableScan converts the DataSource to table scan.
func (ds *DataSource) convertToTableScan(prop *requiredProp) (task task, err error) {
// It will be handled in convertToIndexScan.
if prop.taskTp == copDoubleReadTaskType {
return &copTask{cst: math.MaxFloat64}, nil
}
ts := PhysicalTableScan{
Table: ds.tableInfo,
Columns: ds.Columns,
TableAsName: ds.TableAsName,
DBName: ds.DBName,
}.init(ds.ctx)
ts.SetSchema(ds.schema)
sc := ds.ctx.GetSessionVars().StmtCtx
var pkCol *expression.Column
if ts.Table.PKIsHandle {
if pkColInfo := ts.Table.GetPkColInfo(); pkColInfo != nil {
pkCol = expression.ColInfo2Col(ts.schema.Columns, pkColInfo)
if ds.statisticTable.Columns[pkColInfo.ID] != nil {
ts.Hist = &ds.statisticTable.Columns[pkColInfo.ID].Histogram
}
}
}
if pkCol != nil {
ts.Ranges = ranger.FullIntNewRange(mysql.HasUnsignedFlag(pkCol.RetType.Flag))
} else {
ts.Ranges = ranger.FullIntNewRange(false)
}
statsTbl := ds.statisticTable
rowCount := float64(statsTbl.Count)
if len(ds.pushedDownConds) > 0 {
if pkCol != nil {
ts.AccessCondition, ts.filterCondition = ranger.DetachCondsForTableRange(ds.ctx, ds.pushedDownConds, pkCol)
ts.Ranges, err = ranger.BuildTableRange(ts.AccessCondition, sc, pkCol.RetType)
if err != nil {
return nil, errors.Trace(err)
}
// TODO: We can use ds.getStatsByFilter(accessConditions).
rowCount, err = statsTbl.GetRowCountByIntColumnRanges(sc, pkCol.ID, ts.Ranges)
if err != nil {
return nil, errors.Trace(err)
}
} else {
ts.filterCondition = ds.pushedDownConds
}
}
copTask := &copTask{
tablePlan: ts,
indexPlanFinished: true,
}
task = copTask
matchProperty := len(prop.cols) == 1 && pkCol != nil && prop.cols[0].Equal(nil, pkCol)
// Only use expectedCnt when it's smaller than the count we calculated.
// e.g. IndexScan(count1)->After Filter(count2). The `ds.statsAfterSelect.count` is count2. count1 is the one we need to calculate
// If expectedCnt and count2 are both zero and we go into the below `if` block, the count1 will be set to zero though it's shouldn't be.
if matchProperty && prop.expectedCnt < ds.statsAfterSelect.count {
selectivity, err := statsTbl.Selectivity(ds.ctx, ts.filterCondition)
if err != nil {
log.Warnf("An error happened: %v, we have to use the default selectivity", err.Error())
selectivity = selectionFactor
}
rowCount = math.Min(prop.expectedCnt/selectivity, rowCount)
}
ts.stats = ds.stats.scaleByExpectCnt(rowCount)
copTask.cst = rowCount * scanFactor
if matchProperty {
if prop.desc {
ts.Desc = true
copTask.cst = rowCount * descScanFactor
}
ts.KeepOrder = true
copTask.keepOrder = true
ts.addPushedDownSelection(copTask, ds.statsAfterSelect.scaleByExpectCnt(prop.expectedCnt))
} else {
expectedCnt := math.MaxFloat64
if prop.isEmpty() {
expectedCnt = prop.expectedCnt
} else {
return invalidTask, nil
}
ts.addPushedDownSelection(copTask, ds.statsAfterSelect.scaleByExpectCnt(expectedCnt))
}
if prop.taskTp == rootTaskType {
task = finishCopTask(ds.ctx, task)
} else if _, ok := task.(*rootTask); ok {
return invalidTask, nil
}
return task, nil
}
func (ts *PhysicalTableScan) addPushedDownSelection(copTask *copTask, stats *statsInfo) {
// Add filter condition to table plan now.
if len(ts.filterCondition) > 0 {
sel := PhysicalSelection{Conditions: ts.filterCondition}.init(ts.ctx, stats)
sel.SetChildren(ts)
copTask.tablePlan = sel
// FIXME: It seems wrong...
copTask.cst += copTask.count() * cpuFactor
}
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/pingcap/tidb.git
git@gitee.com:pingcap/tidb.git
pingcap
tidb
tidb
v2.0.1

搜索帮助