org.apache.hadoop.hive.ql.plan.MapJoinDesc Maven / Gradle / Ivy
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hive.ql.plan;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import org.apache.hadoop.hive.conf.HiveConf;
import org.apache.hadoop.hive.ql.plan.Explain.Level;
import org.apache.hadoop.hive.ql.plan.Explain.Vectorization;
import org.apache.hadoop.hive.ql.plan.VectorMapJoinDesc.HashTableImplementationType;
import org.apache.hadoop.hive.ql.plan.VectorMapJoinDesc.OperatorVariation;
/**
* Map Join operator Descriptor implementation.
*
*/
@Explain(displayName = "Map Join Operator", explainLevels = { Level.USER, Level.DEFAULT, Level.EXTENDED })
public class MapJoinDesc extends JoinDesc implements Serializable {
private static final long serialVersionUID = 1L;
private Map> keys;
private TableDesc keyTblDesc;
private List valueTblDescs;
private List valueFilteredTblDescs;
private int posBigTable;
private Map valueIndices;
private Map> retainList;
private transient String bigTableAlias;
// for tez. used to remember which position maps to which logical input
// TODO: should these rather be arrays?
private Map parentToInput = new HashMap();
private Map parentKeyCounts = new HashMap();
private Map parentDataSizes = new HashMap();
// table alias (small) --> input file name (big) --> target file names (small)
private Map>> aliasBucketFileNameMapping;
private Map bigTableBucketNumMapping;
private Map> bigTablePartSpecToFileMapping;
//map join dump file name
private String dumpFilePrefix;
// flag for bucket map join. One usage is to set BucketizedHiveInputFormat
private boolean isBucketMapJoin;
// Hash table memory usage allowed; used in case of non-staged mapjoin.
private float hashtableMemoryUsage; // This is a percentage value between 0 and 1
protected boolean genJoinKeys = true;
private boolean isHybridHashJoin;
private boolean isDynamicPartitionHashJoin = false;
public MapJoinDesc() {
vectorDesc = null;
bigTableBucketNumMapping = new LinkedHashMap();
}
public MapJoinDesc(MapJoinDesc clone) {
super(clone);
if (clone.vectorDesc != null) {
vectorDesc = (VectorDesc) clone.vectorDesc.clone();
}
this.keys = clone.keys;
this.keyTblDesc = clone.keyTblDesc;
this.valueTblDescs = clone.valueTblDescs;
this.posBigTable = clone.posBigTable;
this.valueIndices = clone.valueIndices;
this.retainList = clone.retainList;
this.bigTableAlias = clone.bigTableAlias;
this.aliasBucketFileNameMapping = clone.aliasBucketFileNameMapping;
this.bigTableBucketNumMapping = clone.bigTableBucketNumMapping;
this.bigTablePartSpecToFileMapping = clone.bigTablePartSpecToFileMapping;
this.dumpFilePrefix = clone.dumpFilePrefix;
this.parentToInput = clone.parentToInput;
this.parentKeyCounts = clone.parentKeyCounts;
this.parentDataSizes = clone.parentDataSizes;
this.isBucketMapJoin = clone.isBucketMapJoin;
this.isHybridHashJoin = clone.isHybridHashJoin;
}
public MapJoinDesc(final Map> keys,
final TableDesc keyTblDesc, final Map> values,
final List valueTblDescs,final List valueFilteredTblDescs, List outputColumnNames,
final int posBigTable, final JoinCondDesc[] conds,
final Map> filters, boolean noOuterJoin, String dumpFilePrefix) {
super(values, outputColumnNames, noOuterJoin, conds, filters, null);
vectorDesc = null;
this.keys = keys;
this.keyTblDesc = keyTblDesc;
this.valueTblDescs = valueTblDescs;
this.valueFilteredTblDescs = valueFilteredTblDescs;
this.posBigTable = posBigTable;
this.bigTableBucketNumMapping = new LinkedHashMap();
this.dumpFilePrefix = dumpFilePrefix;
initRetainExprList();
}
private void initRetainExprList() {
retainList = new HashMap>();
Set>> set = super.getExprs().entrySet();
Iterator>> setIter = set.iterator();
while (setIter.hasNext()) {
Entry> current = setIter.next();
List list = new ArrayList();
for (int i = 0; i < current.getValue().size(); i++) {
list.add(i);
}
retainList.put(current.getKey(), list);
}
}
@Explain(displayName = "input vertices", explainLevels = { Level.USER, Level.DEFAULT, Level.EXTENDED })
public Map getParentToInput() {
return parentToInput;
}
public void setParentToInput(Map parentToInput) {
this.parentToInput = parentToInput;
}
public Map getParentKeyCounts() {
return parentKeyCounts;
}
public Map getParentDataSizes() {
return parentDataSizes;
}
@Explain(displayName = "Estimated key counts", explainLevels = { Level.EXTENDED })
public String getKeyCountsExplainDesc() {
StringBuilder result = null;
for (Map.Entry entry : parentKeyCounts.entrySet()) {
if (result == null) {
result = new StringBuilder();
} else {
result.append(", ");
}
result.append(parentToInput.get(entry.getKey())).append(" => ").append(entry.getValue());
}
return result == null ? null : result.toString();
}
public void setParentKeyCount(Map parentKeyCounts) {
this.parentKeyCounts = parentKeyCounts;
}
public Map getValueIndices() {
return valueIndices;
}
public void setValueIndices(Map valueIndices) {
this.valueIndices = valueIndices;
}
public int[] getValueIndex(byte alias) {
return valueIndices == null ? null : valueIndices.get(alias);
}
public Map> getRetainList() {
return retainList;
}
public void setRetainList(Map> retainList) {
this.retainList = retainList;
}
/**
* @return the dumpFilePrefix
*/
public String getDumpFilePrefix() {
return dumpFilePrefix;
}
/**
* @param dumpFilePrefix
* the dumpFilePrefix to set
*/
public void setDumpFilePrefix(String dumpFilePrefix) {
this.dumpFilePrefix = dumpFilePrefix;
}
/**
* @return the keys in string form
*/
@Override
@Explain(displayName = "keys")
public Map getKeysString() {
Map keyMap = new LinkedHashMap();
for (Map.Entry> k: getKeys().entrySet()) {
keyMap.put(k.getKey(), PlanUtils.getExprListString(k.getValue()));
}
return keyMap;
}
@Override
@Explain(displayName = "keys", explainLevels = { Level.USER })
public Map getUserLevelExplainKeysString() {
Map keyMap = new LinkedHashMap();
for (Map.Entry> k: getKeys().entrySet()) {
keyMap.put(k.getKey(), PlanUtils.getExprListString(k.getValue(), true));
}
return keyMap;
}
/**
* @return the keys
*/
public Map> getKeys() {
return keys;
}
/**
* @param keys
* the keys to set
*/
public void setKeys(Map> keys) {
this.keys = keys;
}
/**
* @return the position of the big table not in memory
*/
@Explain(displayName = "Position of Big Table", explainLevels = { Level.EXTENDED })
public int getPosBigTable() {
return posBigTable;
}
/**
* @param posBigTable
* the position of the big table not in memory
*/
public void setPosBigTable(int posBigTable) {
this.posBigTable = posBigTable;
}
/**
* @return the keyTblDesc
*/
public TableDesc getKeyTblDesc() {
return keyTblDesc;
}
/**
* @param keyTblDesc
* the keyTblDesc to set
*/
public void setKeyTblDesc(TableDesc keyTblDesc) {
this.keyTblDesc = keyTblDesc;
}
public List getValueFilteredTblDescs() {
return valueFilteredTblDescs;
}
public void setValueFilteredTblDescs(List valueFilteredTblDescs) {
this.valueFilteredTblDescs = valueFilteredTblDescs;
}
/**
* @return the valueTblDescs
*/
public List getValueTblDescs() {
return valueTblDescs;
}
/**
* @param valueTblDescs
* the valueTblDescs to set
*/
public void setValueTblDescs(List valueTblDescs) {
this.valueTblDescs = valueTblDescs;
}
/**
* @return bigTableAlias
*/
public String getBigTableAlias() {
return bigTableAlias;
}
/**
* @param bigTableAlias
*/
public void setBigTableAlias(String bigTableAlias) {
this.bigTableAlias = bigTableAlias;
}
public Map>> getAliasBucketFileNameMapping() {
return aliasBucketFileNameMapping;
}
public void setAliasBucketFileNameMapping(
Map>> aliasBucketFileNameMapping) {
this.aliasBucketFileNameMapping = aliasBucketFileNameMapping;
}
public Map getBigTableBucketNumMapping() {
return bigTableBucketNumMapping;
}
public void setBigTableBucketNumMapping(Map bigTableBucketNumMapping) {
this.bigTableBucketNumMapping = bigTableBucketNumMapping;
}
public Map> getBigTablePartSpecToFileMapping() {
return bigTablePartSpecToFileMapping;
}
public void setBigTablePartSpecToFileMapping(Map> partToFileMapping) {
this.bigTablePartSpecToFileMapping = partToFileMapping;
}
@Explain(displayName = "BucketMapJoin", explainLevels = { Level.USER, Level.EXTENDED }, displayOnlyOnTrue = true)
public boolean isBucketMapJoin() {
return isBucketMapJoin;
}
public void setBucketMapJoin(boolean isBucketMapJoin) {
this.isBucketMapJoin = isBucketMapJoin;
}
@Explain(displayName = "HybridGraceHashJoin", explainLevels = { Level.USER, Level.DEFAULT, Level.EXTENDED }, displayOnlyOnTrue = true)
public boolean isHybridHashJoin() {
return isHybridHashJoin;
}
public void setHybridHashJoin(boolean isHybridHashJoin) {
this.isHybridHashJoin = isHybridHashJoin;
}
public void setHashTableMemoryUsage(float hashtableMemoryUsage) {
this.hashtableMemoryUsage = hashtableMemoryUsage;
}
public float getHashTableMemoryUsage() {
return hashtableMemoryUsage;
}
@Override
public boolean isMapSideJoin() {
return true;
}
public void setGenJoinKeys(boolean genJoinKeys) {
this.genJoinKeys = genJoinKeys;
}
public boolean getGenJoinKeys() {
return genJoinKeys;
}
public boolean isDynamicPartitionHashJoin() {
return isDynamicPartitionHashJoin;
}
public void setDynamicPartitionHashJoin(boolean isDistributedHashJoin) {
this.isDynamicPartitionHashJoin = isDistributedHashJoin;
}
// Use LinkedHashSet to give predictable display order.
private static final Set vectorizableMapJoinNativeEngines =
new LinkedHashSet(Arrays.asList("tez", "spark"));
public class MapJoinOperatorExplainVectorization extends OperatorExplainVectorization {
private final MapJoinDesc mapJoinDesc;
private final VectorMapJoinDesc vectorMapJoinDesc;
private final VectorMapJoinInfo vectorMapJoinInfo;
private VectorizationCondition[] nativeConditions;
public MapJoinOperatorExplainVectorization(MapJoinDesc mapJoinDesc, VectorDesc vectorDesc) {
// VectorMapJoinOperator is not native vectorized.
super(vectorDesc, ((VectorMapJoinDesc) vectorDesc).hashTableImplementationType() != HashTableImplementationType.NONE);
this.mapJoinDesc = mapJoinDesc;
vectorMapJoinDesc = (VectorMapJoinDesc) vectorDesc;
vectorMapJoinInfo = vectorMapJoinDesc.getVectorMapJoinInfo();
}
private VectorizationCondition[] createNativeConditions() {
boolean enabled = vectorMapJoinDesc.getIsVectorizationMapJoinNativeEnabled();
String engine = vectorMapJoinDesc.getEngine();
String engineInSupportedCondName =
HiveConf.ConfVars.HIVE_EXECUTION_ENGINE.varname + " " + engine + " IN " + vectorizableMapJoinNativeEngines;
boolean engineInSupported = vectorizableMapJoinNativeEngines.contains(engine);
boolean isFastHashTableEnabled = vectorMapJoinDesc.getIsFastHashTableEnabled();
List conditionList = new ArrayList();
conditionList.add(
new VectorizationCondition(
vectorMapJoinDesc.getUseOptimizedTable(),
HiveConf.ConfVars.HIVEMAPJOINUSEOPTIMIZEDTABLE.varname));
conditionList.add(
new VectorizationCondition(
enabled,
HiveConf.ConfVars.HIVE_VECTORIZATION_MAPJOIN_NATIVE_ENABLED.varname));
conditionList.add(
new VectorizationCondition(
engineInSupported,
engineInSupportedCondName));
conditionList.add(
new VectorizationCondition(
vectorMapJoinDesc.getOneMapJoinCondition(),
"One MapJoin Condition"));
conditionList.add(
new VectorizationCondition(
!vectorMapJoinDesc.getHasNullSafes(),
"No nullsafe"));
conditionList.add(
new VectorizationCondition(
vectorMapJoinDesc.getSmallTableExprVectorizes(),
"Small table vectorizes"));
if (isFastHashTableEnabled) {
conditionList.add(
new VectorizationCondition(
!vectorMapJoinDesc.getIsHybridHashJoin(),
"Fast Hash Table and No Hybrid Hash Join"));
} else {
conditionList.add(
new VectorizationCondition(
vectorMapJoinDesc.getSupportsKeyTypes(),
"Optimized Table and Supports Key Types"));
}
VectorizationCondition[] conditions =
conditionList.toArray(new VectorizationCondition[0]);
return conditions;
}
@Explain(vectorization = Vectorization.OPERATOR, displayName = "nativeConditionsMet", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public List getNativeConditionsMet() {
if (nativeConditions == null) {
nativeConditions = createNativeConditions();
}
return VectorizationCondition.getConditionsMet(nativeConditions);
}
@Explain(vectorization = Vectorization.OPERATOR, displayName = "nativeConditionsNotMet", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public List getNativeConditionsNotMet() {
if (nativeConditions == null) {
nativeConditions = createNativeConditions();
}
return VectorizationCondition.getConditionsNotMet(nativeConditions);
}
@Explain(vectorization = Vectorization.EXPRESSION, displayName = "bigTableKeyExpressions", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public List getBigTableKeyExpressions() {
if (!isNative) {
return null;
}
return vectorExpressionsToStringList(vectorMapJoinInfo.getBigTableKeyExpressions());
}
@Explain(vectorization = Vectorization.DETAIL, displayName = "bigTableKeyColumns", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public String getBigTableKeyColumns() {
if (!isNative) {
return null;
}
int[] bigTableKeyColumnMap = vectorMapJoinInfo.getBigTableKeyColumnMap();
if (bigTableKeyColumnMap.length == 0) {
return null;
}
return Arrays.toString(bigTableKeyColumnMap);
}
@Explain(vectorization = Vectorization.EXPRESSION, displayName = "bigTableValueExpressions", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public List getBigTableValueExpressions() {
if (!isNative) {
return null;
}
return vectorExpressionsToStringList(vectorMapJoinInfo.getBigTableValueExpressions());
}
@Explain(vectorization = Vectorization.DETAIL, displayName = "bigTableValueColumns", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public String getBigTableValueColumns() {
if (!isNative) {
return null;
}
int[] bigTableValueColumnMap = vectorMapJoinInfo.getBigTableValueColumnMap();
if (bigTableValueColumnMap.length == 0) {
return null;
}
return Arrays.toString(bigTableValueColumnMap);
}
@Explain(vectorization = Vectorization.DETAIL, displayName = "smallTableMapping", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public String getSmallTableColumns() {
if (!isNative) {
return null;
}
return outputColumnsToStringList(vectorMapJoinInfo.getSmallTableMapping());
}
@Explain(vectorization = Vectorization.DETAIL, displayName = "projectedOutputColumns", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public String getProjectedOutputColumns() {
if (!isNative) {
return null;
}
return outputColumnsToStringList(vectorMapJoinInfo.getProjectionMapping());
}
@Explain(vectorization = Vectorization.DETAIL, displayName = "bigTableOuterKeyMapping", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public List getBigTableOuterKey() {
if (!isNative || vectorMapJoinDesc.operatorVariation() != OperatorVariation.OUTER) {
return null;
}
return columnMappingToStringList(vectorMapJoinInfo.getBigTableOuterKeyMapping());
}
@Explain(vectorization = Vectorization.DETAIL, displayName = "bigTableRetainedColumns", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public String getBigTableRetainedColumns() {
if (!isNative) {
return null;
}
return outputColumnsToStringList(vectorMapJoinInfo.getBigTableRetainedMapping());
}
@Explain(vectorization = Vectorization.OPERATOR, displayName = "nativeNotSupportedKeyTypes", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public List getNativeNotSupportedKeyTypes() {
return vectorMapJoinDesc.getNotSupportedKeyTypes();
}
}
@Explain(vectorization = Vectorization.OPERATOR, displayName = "Map Join Vectorization", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public MapJoinOperatorExplainVectorization getMapJoinVectorization() {
if (vectorDesc == null || this instanceof SMBJoinDesc) {
return null;
}
return new MapJoinOperatorExplainVectorization(this, vectorDesc);
}
public class SMBJoinOperatorExplainVectorization extends OperatorExplainVectorization {
private final SMBJoinDesc smbJoinDesc;
private final VectorSMBJoinDesc vectorSMBJoinDesc;
public SMBJoinOperatorExplainVectorization(SMBJoinDesc smbJoinDesc, VectorDesc vectorDesc) {
// Native vectorization NOT supported.
super(vectorDesc, false);
this.smbJoinDesc = smbJoinDesc;
vectorSMBJoinDesc = (VectorSMBJoinDesc) vectorDesc;
}
}
// Handle dual nature.
@Explain(vectorization = Vectorization.OPERATOR, displayName = "SMB Map Join Vectorization", explainLevels = { Level.DEFAULT, Level.EXTENDED })
public SMBJoinOperatorExplainVectorization getSMBJoinVectorization() {
if (vectorDesc == null || !(this instanceof SMBJoinDesc)) {
return null;
}
return new SMBJoinOperatorExplainVectorization((SMBJoinDesc) this, vectorDesc);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy