//--------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner [....] // @backupOwner [....] //--------------------------------------------------------------------- using System.Collections.Generic; using System.Data.Common; using System.Data.Metadata.Edm; using System.Data.Query.PlanCompiler; using System.Diagnostics; using System.Linq; namespace System.Data.Query.InternalTrees { /// /// The Command object encapsulates all information relating to a single command. /// It includes the expression tree in question, as well as the parameters to the /// command. /// Additionally, the Command class serves as a factory for building up different /// nodes and Ops. Every node in the tree has a unique id, and this is enforced by /// the node factory methods /// internal class Command { #region private state private Dictionary m_parameterMap; private List m_vars; private List m_tables; private Node m_root; private MetadataWorkspace m_metadataWorkspace; private TypeUsage m_boolType; private TypeUsage m_intType; private TypeUsage m_stringType; private ConstantPredicateOp m_trueOp; private ConstantPredicateOp m_falseOp; private NodeInfoVisitor m_nodeInfoVisitor; private PlanCompiler.KeyPullup m_keyPullupVisitor; private int m_nextNodeId; private int m_nextBranchDiscriminatorValue = 1000; private bool m_disableVarVecEnumCaching; private Stack m_freeVarVecEnumerators; private Stack m_freeVarVecs; // set of referenced rel properties in this query private HashSet m_referencedRelProperties; #endregion #region constructors /// /// Creates a new command /// internal Command(MetadataWorkspace metadataWorkspace) { m_parameterMap = new Dictionary(); m_vars = new List(); m_tables = new List
(); m_metadataWorkspace = metadataWorkspace; if(!TryGetPrimitiveType(PrimitiveTypeKind.Boolean, out m_boolType)) { throw EntityUtil.ProviderIncompatible(System.Data.Entity.Strings.Cqt_General_NoProviderBooleanType); } if (!TryGetPrimitiveType(PrimitiveTypeKind.Int32, out m_intType)) { throw EntityUtil.ProviderIncompatible(System.Data.Entity.Strings.Cqt_General_NoProviderIntegerType); } if (!TryGetPrimitiveType(PrimitiveTypeKind.String, out m_stringType)) { throw EntityUtil.ProviderIncompatible(System.Data.Entity.Strings.Cqt_General_NoProviderStringType); } m_trueOp = new ConstantPredicateOp(m_boolType, true); m_falseOp = new ConstantPredicateOp(m_boolType, false); m_nodeInfoVisitor = new NodeInfoVisitor(this); m_keyPullupVisitor = new PlanCompiler.KeyPullup(this); // FreeLists m_freeVarVecEnumerators = new Stack(); m_freeVarVecs = new Stack(); m_referencedRelProperties = new HashSet(); } #endregion #region public methods /// /// Gets the metadata workspace associated with this command /// internal MetadataWorkspace MetadataWorkspace { get { return m_metadataWorkspace; } } /// /// Gets/sets the root node of the query /// internal Node Root { get { return m_root; } set { m_root = value; } } internal void DisableVarVecEnumCaching() { m_disableVarVecEnumCaching = true; } /// /// Returns the next value for a UnionAll BranchDiscriminator. /// internal int NextBranchDiscriminatorValue { get { return m_nextBranchDiscriminatorValue++; } } /// /// Returns the next value for a node id, without incrementing it. /// internal int NextNodeId { get { return m_nextNodeId; } } #region Metadata Helpers /// /// Helper routine to get the metadata representation for the bool type /// internal TypeUsage BooleanType { get { return m_boolType; } } /// /// Helper routine to get the metadata representation of the int type /// internal TypeUsage IntegerType { get { return m_intType; } } /// /// Helper routine to get the metadata representation of the string type /// internal TypeUsage StringType { get { return m_stringType; } } /// /// Get the primitive type by primitive type kind /// /// EdmMetadata.PrimitiveTypeKind of the primitive type /// A TypeUsage that represents the specified primitive type /// True if the specified primitive type could be retrieved; otherwise false. private bool TryGetPrimitiveType(PrimitiveTypeKind modelType, out TypeUsage type) { type = null; if (modelType == PrimitiveTypeKind.String) { type = TypeUsage.CreateStringTypeUsage(m_metadataWorkspace.GetModelPrimitiveType(modelType), false /*unicode*/, false /*fixed*/); } else { type = m_metadataWorkspace.GetCanonicalModelTypeUsage(modelType); } return (null != type); } #endregion #region VarVec Creation /// /// VarVec constructor /// /// A new, empty, VarVec internal VarVec CreateVarVec() { VarVec vec; if (m_freeVarVecs.Count == 0) { vec = new VarVec(this); } else { vec = m_freeVarVecs.Pop(); vec.Clear(); } return vec; } /// /// Create a VarVec with a single Var /// /// /// internal VarVec CreateVarVec(Var v) { VarVec varset = CreateVarVec(); varset.Set(v); return varset; } /// /// Create a VarVec with the set of specified vars /// /// /// internal VarVec CreateVarVec(IEnumerable v) { VarVec vec = CreateVarVec(); vec.InitFrom(v); return vec; } /// /// Create a new VarVec from the input VarVec /// /// /// internal VarVec CreateVarVec(VarVec v) { VarVec vec = CreateVarVec(); vec.InitFrom(v); return vec; } /// /// Release a VarVec to the freelist /// /// internal void ReleaseVarVec(VarVec vec) { m_freeVarVecs.Push(vec); } #endregion #region VarVecEnumerator /// /// Create a new enumerator for a VarVec; use a free one if its /// available; otherwise, create a new one /// /// /// internal VarVec.VarVecEnumerator GetVarVecEnumerator(VarVec vec) { VarVec.VarVecEnumerator enumerator; if (m_disableVarVecEnumCaching || m_freeVarVecEnumerators.Count == 0) { enumerator = new VarVec.VarVecEnumerator(vec); } else { enumerator = m_freeVarVecEnumerators.Pop(); enumerator.Init(vec); } return enumerator; } /// /// Release an enumerator; keep it in a local stack for future use /// /// internal void ReleaseVarVecEnumerator(VarVec.VarVecEnumerator enumerator) { if (!m_disableVarVecEnumCaching) { m_freeVarVecEnumerators.Push(enumerator); } } #endregion #region VarList /// /// Create an ordered list of Vars - initially empty /// /// internal static VarList CreateVarList() { return new VarList(); } /// /// Create an ordered list of Vars /// /// /// internal static VarList CreateVarList(IEnumerable vars) { return new VarList(vars); } #endregion #region VarMap internal VarMap CreateVarMap() { return new VarMap(); } #endregion #region Table Helpers private int NewTableId() { return m_tables.Count; } /// /// Create a table whose element type is "elementType" /// /// type of each element (row) of the table /// a table definition object internal static TableMD CreateTableDefinition(TypeUsage elementType) { return new TableMD(elementType, null); } /// /// Creates a new table definition based on an extent. The element type /// of the extent manifests as the single column of the table /// /// the metadata extent /// A new TableMD instance based on the extent internal static TableMD CreateTableDefinition(EntitySetBase extent) { return new TableMD(TypeUsage.Create(extent.ElementType), extent); } /// /// Create a "flat" table definition object (ie) the table has one column /// for each property of the specified row type /// /// the shape of each row of the table /// the table definition internal TableMD CreateFlatTableDefinition(RowType type) { return CreateFlatTableDefinition(type.Properties, new List(), null); } /// /// Create a "flat" table defintion. The table has one column for each property /// specified, and the key columns of the table are those specified in the /// keyMembers parameter /// /// list of columns for the table /// the key columns (if any) /// (OPTIONAL) entityset corresponding to this table /// internal TableMD CreateFlatTableDefinition(IEnumerable properties, IEnumerable keyMembers, EntitySetBase entitySet) { return new TableMD(properties, keyMembers, entitySet); } /// /// Creates a new table instance /// /// table metadata /// A new Table instance with columns as defined in the specified metadata internal Table CreateTableInstance(TableMD tableMetadata) { Table t = new Table(this, tableMetadata, NewTableId()); m_tables.Add(t); return t; } #endregion #region Var Access /// /// All vars in the query /// internal IEnumerable Vars { get { return m_vars.Where(v => v.VarType != VarType.NotValid); } } /// /// Access an existing variable in the query (by its id) /// /// The ID of the variable to retrieve /// The variable with the specified ID internal Var GetVar(int id) { Debug.Assert(m_vars[id].VarType != VarType.NotValid, "The var has been replaced by a different var and is no longer valid."); return m_vars[id]; } /// /// Gets the ParameterVar that corresponds to a given named parameter /// /// The name of the parameter for which to retrieve the ParameterVar /// The ParameterVar that corresponds to the specified parameter internal ParameterVar GetParameter(string paramName) { return m_parameterMap[paramName]; } #endregion #region Var Creation private int NewVarId() { return m_vars.Count; } /// /// Creates a variable for a parameter in the query /// /// The name of the parameter for which to create the var /// The type of the parameter, and therefore the new var /// A new ParameterVar instance with the specified name and type internal ParameterVar CreateParameterVar(string parameterName, TypeUsage parameterType) { if (m_parameterMap.ContainsKey(parameterName)) throw new Exception("duplicate parameter name: " + parameterName); ParameterVar v = new ParameterVar(NewVarId(), parameterType, parameterName); m_vars.Add(v); m_parameterMap[parameterName] = v; return v; } /// /// Creates a variable for the given parameter variable and replaces it in parameter map. /// /// Parameter variable that needs to replaced. /// Delegate that generates the replacement parameter's type. /// A new ParameterVar instance created of . /// /// This method should be used only to replace external enum or strong spatial parameters with a counterpart whose /// type is the underlying type of the enum type, or the union type contating the strong spatial type of the . /// The operation invalidates the . After the operation has completed /// the ) is invalidated internally and should no longer be used. /// Func< private ParameterVar ReplaceParameterVar(ParameterVar oldVar, Func generateReplacementType) { Debug.Assert(oldVar != null, "oldVar != null"); Debug.Assert(m_vars.Contains(oldVar)); ParameterVar v = new ParameterVar(NewVarId(), generateReplacementType(oldVar.Type), oldVar.ParameterName); m_parameterMap[oldVar.ParameterName] = v; m_vars.Add(v); return v; } /// /// Creates a variable for the given enum parameter variable and replaces it in parameter map. /// /// Enum parameter variable that needs to replaced. /// A new ParameterVar instance created of . /// /// This method should be used only to replace external enum parameter with a counterpart whose /// type is the underlying type of the enum type of the . /// The operation invalidates the . After the operation has completed /// the ) is invalidated internally and should no longer be used. /// internal ParameterVar ReplaceEnumParameterVar(ParameterVar oldVar) { return ReplaceParameterVar(oldVar, t => TypeHelpers.CreateEnumUnderlyingTypeUsage(t)); } /// /// Creates a variable for the given spatial parameter variable and replaces it in parameter map. /// /// Spatial parameter variable that needs to replaced. /// A new ParameterVar instance created of . /// /// This method should be used only to replace external strong spatial parameter with a counterpart whose /// type is the appropriate union type for . /// The operation invalidates the . After the operation has completed /// the ) is invalidated internally and should no longer be used. /// internal ParameterVar ReplaceStrongSpatialParameterVar(ParameterVar oldVar) { return ReplaceParameterVar(oldVar, t => TypeHelpers.CreateSpatialUnionTypeUsage(t)); } /// /// Creates a new var for a table column /// /// The table instance that produces the column /// column metadata /// A new ColumnVar instance that references the specified column in the given table internal ColumnVar CreateColumnVar(Table table, ColumnMD columnMD) { // create a new column var now ColumnVar c = new ColumnVar(NewVarId(), table, columnMD); table.Columns.Add(c); m_vars.Add(c); return c; } /// /// Creates a computed var (ie) a variable that is computed by an expression /// /// The type of the result produced by the expression that defines the variable /// A new ComputedVar instance with the specified result type internal ComputedVar CreateComputedVar(TypeUsage type) { ComputedVar v = new ComputedVar(NewVarId(), type); m_vars.Add(v); return v; } /// /// Creates a SetOp Var of /// /// Datatype of the Var /// A new SetOp Var with the specified result type internal SetOpVar CreateSetOpVar(TypeUsage type) { SetOpVar v = new SetOpVar(NewVarId(), type); m_vars.Add(v); return v; } #endregion #region Node Creation // // The routines below help in node construction. All command tree nodes must go // through these routines. These routines help to stamp each node with a unique // id (the id is very helpful for debugging) // /// /// Creates a Node with zero children /// /// The operator that the Node should reference /// A new Node with zero children that references the specified Op internal Node CreateNode(Op op) { return this.CreateNode(op, new List()); } /// /// Creates a node with a single child Node /// /// The operator that the Node should reference /// The single child Node /// A new Node with the specified child Node, that references the specified Op internal Node CreateNode(Op op, Node arg1) { List l = new List(); l.Add(arg1); return this.CreateNode(op, l); } /// /// Creates a node with two child Nodes /// /// The operator that the Node should reference /// The first child Node /// the second child Node /// A new Node with the specified child Nodes, that references the specified Op internal Node CreateNode(Op op, Node arg1, Node arg2) { List l = new List(); l.Add(arg1); l.Add(arg2); return this.CreateNode(op, l); } /// /// Creates a node with 3 child Nodes /// /// The operator that the Node should reference /// The first child Node /// The second child Node /// The third child Node /// A new Node with the specified child Nodes, that references the specified Op internal Node CreateNode(Op op, Node arg1, Node arg2, Node arg3) { List l = new List(); l.Add(arg1); l.Add(arg2); l.Add(arg3); return this.CreateNode(op, l); } /// /// Create a Node with the specified list of child Nodes /// /// The operator that the Node should reference /// The list of child Nodes /// A new Node with the specified child nodes, that references the specified Op internal Node CreateNode(Op op, IList args) { return new Node(m_nextNodeId++, op, new List(args)); } /// /// Create a Node with the specified list of child Nodes /// /// The operator that the Node should reference /// The list of child Nodes /// A new Node with the specified child nodes, that references the specified Op internal Node CreateNode(Op op, List args) { return new Node(m_nextNodeId++, op, args); } #endregion #region ScalarOps /// /// Creates a new ConstantOp /// /// The type of the constant value /// The constant value (may be null) /// A new ConstantOp with the specified type and value internal ConstantBaseOp CreateConstantOp(TypeUsage type, object value) { // create a NullOp if necessary if (value == null) { return new NullOp(type); } // Identify "safe" constants - the only safe ones are boolean (and we should // probably include ints eventually) else if (TypeSemantics.IsBooleanType(type)) { return new InternalConstantOp(type, value); } else { return new ConstantOp(type, value); } } /// /// Create an "internal" constantOp - only for use by the plan compiler to /// represent internally generated constants. /// User constants in the query should never get into this function /// /// datatype of the constant /// constant value /// a new "internal" constant op that represents the constant internal InternalConstantOp CreateInternalConstantOp(TypeUsage type, object value) { return new InternalConstantOp(type, value); } /// /// An internal constant that serves as a null sentinel, i.e. it is only ever used /// to be checked whether it is null /// /// internal NullSentinelOp CreateNullSentinelOp() { return new NullSentinelOp(this.IntegerType, 1); } /// /// An "internal" null constant /// /// datatype of the null constant /// a new "internal" null constant op internal NullOp CreateNullOp(TypeUsage type) { return new NullOp(type); } /// /// Create a constant predicateOp /// /// value of the constant predicate /// internal ConstantPredicateOp CreateConstantPredicateOp(bool value) { return value ? m_trueOp : m_falseOp; } /// /// Create a constant predicate with value=true /// /// internal ConstantPredicateOp CreateTrueOp() { return m_trueOp; } /// /// Create a constant predicateOp with the value false /// /// internal ConstantPredicateOp CreateFalseOp() { return m_falseOp; } /// /// Creates a new FunctionOp /// /// EdmFunction metadata that represents the function that is invoked by the Op /// A new FunctionOp that references the specified function metadata internal FunctionOp CreateFunctionOp(EdmFunction function) { return new FunctionOp(function); } /// /// Creates a new TreatOp /// /// Type metadata that specifies the type that the child of the treat node should be treated as /// A new TreatOp that references the specified type metadata internal TreatOp CreateTreatOp(TypeUsage type) { return new TreatOp(type, false); } /// /// Create a "dummy" treatOp (i.e.) we can actually ignore the treatOp. /// /// /// internal TreatOp CreateFakeTreatOp(TypeUsage type) { return new TreatOp(type, true); } /// /// Creates a new IsOfOp, which tests if the argument is of the specified type or a promotable type /// /// Type metadata that specifies the type with which the type of the argument should be compared /// A new IsOfOp that references the specified type metadata internal IsOfOp CreateIsOfOp(TypeUsage isOfType) { return new IsOfOp(isOfType, false/*only*/, m_boolType); } /// /// Creates a new IsOfOp, which tests if the argument is of the specified type (and only the specified type) /// /// Type metadata that specifies the type with which the type of the argument should be compared /// A new IsOfOp that references the specified type metadata internal IsOfOp CreateIsOfOnlyOp(TypeUsage isOfType) { return new IsOfOp(isOfType, true /* "only" */, m_boolType); } /// /// Creates a new CastOp /// /// Type metadata that represents the type to which the argument should be cast /// A new CastOp that references the specified type metadata internal CastOp CreateCastOp(TypeUsage type) { return new CastOp(type); } /// /// Creates a new SoftCastOp and casts the input to the desired type. /// /// The caller is expected to determine if the cast is necessary or not /// /// Type metadata that represents the type to which the argument should be cast /// A new CastOp that references the specified type metadata internal SoftCastOp CreateSoftCastOp(TypeUsage type) { return new SoftCastOp(type); } /// /// Creates a new ComparisonOp of the specified type /// /// An OpType that specifies one of the valid comparison OpTypes: EQ, GT, GE, NE, LT, LE /// A new ComparisonOp of the specified comparison OpType internal ComparisonOp CreateComparisonOp(OpType opType) { return new ComparisonOp(opType, this.BooleanType); } /// /// Creates a new LikeOp /// /// The new LikeOp internal LikeOp CreateLikeOp() { return new LikeOp(this.BooleanType); } /// /// Creates a new ConditionalOp of the specified type /// /// An OpType that specifies one of the valid condition operations: And, Or, Not, IsNull /// A new ConditionalOp with the specified conditional OpType internal ConditionalOp CreateConditionalOp(OpType opType) { return new ConditionalOp(opType, this.BooleanType); } /// /// Creates a new CaseOp /// /// The result type of the CaseOp /// A new CaseOp with the specified result type internal CaseOp CreateCaseOp(TypeUsage type) { return new CaseOp(type); } /// /// Creates a new AggregateOp /// /// EdmFunction metadata that specifies the aggregate function /// Indicates whether or not the aggregate is a distinct aggregate /// A new AggregateOp with the specified function metadata and distinct property internal AggregateOp CreateAggregateOp(EdmFunction aggFunc, bool distinctAgg) { return new AggregateOp(aggFunc, distinctAgg); } /// /// Creates a named type constructor /// /// Type metadata that specifies the type of the instance to construct /// A new NewInstanceOp with the specified result type internal NewInstanceOp CreateNewInstanceOp(TypeUsage type) { return new NewInstanceOp(type); } /// /// Build out a new NewEntityOp constructing the entity scoped to the . /// internal NewEntityOp CreateScopedNewEntityOp(TypeUsage type, List relProperties, EntitySet entitySet) { return new NewEntityOp(type, relProperties, true, entitySet); } /// /// Build out a new NewEntityOp constructing the uscoped entity . /// internal NewEntityOp CreateNewEntityOp(TypeUsage type, List relProperties) { return new NewEntityOp(type, relProperties, false, null); } /// /// Create a discriminated named type constructor /// /// Type metadata that specifies the type of the instance to construct /// Mapping information including discriminator values /// the entityset that this instance belongs to /// list of rel properties that have corresponding values /// A new DiscriminatedNewInstanceOp with the specified result type and discrimination behavior internal DiscriminatedNewEntityOp CreateDiscriminatedNewEntityOp(TypeUsage type, ExplicitDiscriminatorMap discriminatorMap, EntitySet entitySet, List relProperties) { return new DiscriminatedNewEntityOp(type, discriminatorMap, entitySet, relProperties); } /// /// Creates a multiset constructor /// /// Type metadata that specifies the type of the multiset to construct /// A new NewMultiSetOp with the specified result type internal NewMultisetOp CreateNewMultisetOp(TypeUsage type) { return new NewMultisetOp(type); } /// /// Creates a record constructor /// /// Type metadata that specifies that record type to construct /// A new NewRecordOp with the specified result type internal NewRecordOp CreateNewRecordOp(TypeUsage type) { return new NewRecordOp(type); } /// /// Creates a record constructor /// /// Type metadata that specifies that record type to construct /// A new NewRecordOp with the specified result type internal NewRecordOp CreateNewRecordOp(RowType type) { return new NewRecordOp(TypeUsage.Create(type)); } /// /// A variant of the above method to create a NewRecordOp. An additional /// argument - fields - is supplied, and the semantics is that only these fields /// have any values specified as part of the Node. All other fields are /// considered to be null. /// /// /// /// internal NewRecordOp CreateNewRecordOp(TypeUsage type, List fields) { return new NewRecordOp(type, fields); } /// /// Creates a new VarRefOp /// /// The variable to reference /// A new VarRefOp that references the specified variable internal VarRefOp CreateVarRefOp(Var v) { return new VarRefOp(v); } /// /// Creates a new ArithmeticOp of the specified type /// /// An OpType that specifies one of the valid arithmetic operations: Plus, Minus, Multiply, Divide, Modulo, UnaryMinus /// Type metadata that specifies the result type of the arithmetic operation /// A new ArithmeticOp of the specified arithmetic OpType internal ArithmeticOp CreateArithmeticOp(OpType opType, TypeUsage type) { return new ArithmeticOp(opType, type); } /// /// Creates a new PropertyOp /// /// EdmProperty metadata that specifies the property /// A new PropertyOp that references the specified property metadata internal PropertyOp CreatePropertyOp(EdmMember prop) { // // Track all rel-properties // NavigationProperty navProp = prop as NavigationProperty; if (navProp != null) { RelProperty relProperty = new RelProperty(navProp.RelationshipType, navProp.FromEndMember, navProp.ToEndMember); AddRelPropertyReference(relProperty); RelProperty inverseRelProperty = new RelProperty(navProp.RelationshipType, navProp.ToEndMember, navProp.FromEndMember); AddRelPropertyReference(inverseRelProperty); } // Actually create the propertyOp return new PropertyOp(Helper.GetModelTypeUsage(prop), prop); } /// /// Create a "relationship" propertyOp /// /// the relationship property /// a RelPropertyOp internal RelPropertyOp CreateRelPropertyOp(RelProperty prop) { AddRelPropertyReference(prop); return new RelPropertyOp(prop.ToEnd.TypeUsage, prop); } /// /// Creates a new RefOp /// /// The EntitySet to which the ref refers /// The result type of the RefOp /// A new RefOp that references the specified EntitySet and has the specified result type internal RefOp CreateRefOp(EntitySet entitySet, TypeUsage type) { return new RefOp(entitySet, type); } /// /// Creates a new ExistsOp /// /// A new ExistsOp internal ExistsOp CreateExistsOp() { return new ExistsOp(this.BooleanType); } /// /// Creates a new ElementOp /// /// Type metadata that specifies the result (element) type /// A new ElementOp with the specified result type internal ElementOp CreateElementOp(TypeUsage type) { return new ElementOp(type); } /// /// Creates a new GetEntityRefOp: a ref-extractor (from an entity instance) Op /// /// Type metadata that specifies the result type /// A new GetEntityKeyOp with the specified result type internal GetEntityRefOp CreateGetEntityRefOp(TypeUsage type) { return new GetEntityRefOp(type); } /// /// Creates a new GetRefKeyOp: a key-extractor (from a ref instance) Op /// /// Type metadata that specifies the result type /// A new GetRefKeyOp with the specified result type internal GetRefKeyOp CreateGetRefKeyOp(TypeUsage type) { return new GetRefKeyOp(type); } /// /// Creates a new CollectOp /// /// Type metadata that specifies the result type of the Nest operation /// A new NestOp with the specified result type internal CollectOp CreateCollectOp(TypeUsage type) { return new CollectOp(type); } /// /// Create a DerefOp /// /// Entity type of the target entity /// a DerefOp internal DerefOp CreateDerefOp(TypeUsage type) { return new DerefOp(type); } /// /// Create a new NavigateOp node /// /// the output type of the navigateOp /// the relationship property /// the navigateOp internal NavigateOp CreateNavigateOp(TypeUsage type, RelProperty relProperty) { // keep track of rel-properties AddRelPropertyReference(relProperty); return new NavigateOp(type, relProperty); } #endregion #region AncillaryOps /// /// Creates a VarDefListOp /// /// A new VarDefListOp internal VarDefListOp CreateVarDefListOp() { return VarDefListOp.Instance; } /// /// Creates a VarDefOp (for a computed var) /// /// The computed var /// A new VarDefOp that references the computed var internal VarDefOp CreateVarDefOp(Var v) { return new VarDefOp(v); } /// /// Create a VarDefOp and the associated node for an expression. /// We create a computedVar first - of the same type as the expression, and /// then create a VarDefOp for the computed Var. Finally, we create a Node for /// the VarDefOp /// /// /// new Var produced /// internal Node CreateVarDefNode(Node definingExpr, out Var computedVar) { Debug.Assert(definingExpr.Op != null); ScalarOp scalarOp = definingExpr.Op as ScalarOp; Debug.Assert(scalarOp != null); computedVar = this.CreateComputedVar(scalarOp.Type); VarDefOp varDefOp = this.CreateVarDefOp(computedVar); Node varDefNode = this.CreateNode(varDefOp, definingExpr); return varDefNode; } /// /// Creates a VarDefListOp with a single child - a VarDefOp created as in the function /// above. /// /// /// the computed Var produced /// internal Node CreateVarDefListNode(Node definingExpr, out Var computedVar) { Node varDefNode = this.CreateVarDefNode(definingExpr, out computedVar); VarDefListOp op = this.CreateVarDefListOp(); Node varDefListNode = this.CreateNode(op, varDefNode); return varDefListNode; } #endregion #region RelOps /// /// Creates a new ScanTableOp /// /// A Table metadata instance that specifies the table that should be scanned /// A new ScanTableOp that references a new Table instance based on the specified table metadata internal ScanTableOp CreateScanTableOp(TableMD tableMetadata) { Table table = this.CreateTableInstance(tableMetadata); return CreateScanTableOp(table); } /// /// A variant of the above /// /// The table instance /// a new ScanTableOp internal ScanTableOp CreateScanTableOp(Table table) { return new ScanTableOp(table); } /// /// Creates an instance of a ScanViewOp /// /// the table instance /// a new ScanViewOp internal ScanViewOp CreateScanViewOp(Table table) { return new ScanViewOp(table); } /// /// Creates an instance of a ScanViewOp /// /// the table metadata /// a new ScanViewOp internal ScanViewOp CreateScanViewOp(TableMD tableMetadata) { Table table = this.CreateTableInstance(tableMetadata); return this.CreateScanViewOp(table); } /// /// Creates a new UnnestOp, which creates a streaming result from a scalar (non-RelOp) value /// /// The Var that indicates the value to unnest /// A new UnnestOp that targets the specified Var internal UnnestOp CreateUnnestOp(Var v) { Table t = this.CreateTableInstance(Command.CreateTableDefinition(TypeHelpers.GetEdmType(v.Type).TypeUsage)); return CreateUnnestOp(v, t); } /// /// Creates a new UnnestOp - a variant of the above with the Table supplied /// /// the unnest Var /// the table instance /// a new UnnestOp internal UnnestOp CreateUnnestOp(Var v, Table t) { return new UnnestOp(v, t); } /// /// Creates a new FilterOp /// /// A new FilterOp internal FilterOp CreateFilterOp() { return FilterOp.Instance; } /// /// Creates a new ProjectOp /// /// A VarSet that specifies the Vars produced by the projection /// A new ProjectOp with the specified output VarSet internal ProjectOp CreateProjectOp(VarVec vars) { return new ProjectOp(vars); } /// /// A variant of the above where the ProjectOp produces exactly one var /// /// /// internal ProjectOp CreateProjectOp(Var v) { VarVec varSet = this.CreateVarVec(); varSet.Set(v); return new ProjectOp(varSet); } #region JoinOps /// /// Creates a new InnerJoinOp /// /// A new InnerJoinOp internal InnerJoinOp CreateInnerJoinOp() { return InnerJoinOp.Instance; } /// /// Creates a new LeftOuterJoinOp /// /// A new LeftOuterJoinOp internal LeftOuterJoinOp CreateLeftOuterJoinOp() { return LeftOuterJoinOp.Instance; } /// /// Creates a new FullOuterJoinOp /// /// A new FullOuterJoinOp internal FullOuterJoinOp CreateFullOuterJoinOp() { return FullOuterJoinOp.Instance; } /// /// Creates a new CrossJoinOp /// /// A new CrossJoinOp internal CrossJoinOp CreateCrossJoinOp() { return CrossJoinOp.Instance; } #endregion #region ApplyOps /// /// Creates a new CrossApplyOp /// /// A new CrossApplyOp internal CrossApplyOp CreateCrossApplyOp() { return CrossApplyOp.Instance; } /// /// Creates a new OuterApplyOp /// /// A new OuterApplyOp internal OuterApplyOp CreateOuterApplyOp() { return OuterApplyOp.Instance; } #endregion #region SortKeys /// /// Creates a new SortKey with the specified var, order and collation /// /// The variable to sort on /// The sort order (true for ascending, false for descending) /// The sort collation /// A new SortKey with the specified var, order and collation internal static SortKey CreateSortKey(Var v, bool asc, string collation) { return new SortKey(v, asc, collation); } /// /// Creates a new SortKey with the specified var and order /// /// The variable to sort on /// The sort order (true for ascending, false for descending) /// A new SortKey with the specified var and order internal static SortKey CreateSortKey(Var v, bool asc) { return new SortKey(v, asc, ""); } /// /// Creates a new SortKey with the specified var /// /// The variable to sort on /// A new SortKey with the specified var internal static SortKey CreateSortKey(Var v) { return new SortKey(v, true, ""); } #endregion /// /// Creates a new SortOp /// /// The list of SortKeys that define the sort var, order and collation for each sort key /// A new SortOp with the specified sort keys internal SortOp CreateSortOp(List sortKeys) { return new SortOp(sortKeys); } /// /// Creates a new ConstrainedSortOp /// /// The list of SortKeys that define the sort var, order and collation for each sort key /// A new ConstrainedSortOp with the specified sort keys and a default WithTies value of false internal ConstrainedSortOp CreateConstrainedSortOp(List sortKeys) { return new ConstrainedSortOp(sortKeys, false); } /// /// Creates a new ConstrainedSortOp /// /// The list of SortKeys that define the sort var, order and collation for each sort key /// The value to use for the WithTies property of the new ConstrainedSortOp /// A new ConstrainedSortOp with the specified sort keys and WithTies value internal ConstrainedSortOp CreateConstrainedSortOp(List sortKeys, bool withTies) { return new ConstrainedSortOp(sortKeys, withTies); } /// /// Creates a new GroupByOp /// /// A VarSet that specifies the Key variables produced by the GroupByOp /// A VarSet that specifies all (Key and Aggregate) variables produced by the GroupByOp /// A new GroupByOp with the specified key and output VarSets internal GroupByOp CreateGroupByOp(VarVec gbyKeys, VarVec outputs) { return new GroupByOp(gbyKeys, outputs); } /// /// Creates a new GroupByIntoOp /// /// A VarSet that specifies the Key variables produced by the GroupByOp /// A VarSet that specifies the vars from the input that represent the real grouping input /// A VarSet that specifies all (Key and Aggregate) variables produced by the GroupByOp /// A new GroupByOp with the specified key and output VarSets internal GroupByIntoOp CreateGroupByIntoOp(VarVec gbyKeys, VarVec inputs, VarVec outputs) { return new GroupByIntoOp(gbyKeys, inputs, outputs); } /// /// Creates a new DistinctOp /// list of key vars /// /// A new DistinctOp internal DistinctOp CreateDistinctOp(VarVec keyVars) { return new DistinctOp(keyVars); } /// /// An overload of the above - where the distinct has exactly one key /// /// /// internal DistinctOp CreateDistinctOp(Var keyVar) { return new DistinctOp(this.CreateVarVec(keyVar)); } /// /// Creates a new UnionAllOp /// /// Mappings from the Output Vars to the Vars produced by the left argument /// Mappings from the Output Vars to the Vars produced by the right argument /// A UnionAllOp that references the specified left and right Vars internal UnionAllOp CreateUnionAllOp(VarMap leftMap, VarMap rightMap) { return CreateUnionAllOp(leftMap, rightMap, null); } /// /// Creates a new UnionAllOp, with a branch descriminator. /// /// Mappings from the Output Vars to the Vars produced by the left argument /// Mappings from the Output Vars to the Vars produced by the right argument /// Var that contains the branch discrimination value (may be null until key pullup occurs) /// A UnionAllOp that references the specified left and right Vars internal UnionAllOp CreateUnionAllOp(VarMap leftMap, VarMap rightMap, Var branchDiscriminator) { Debug.Assert(leftMap.Count == rightMap.Count, "VarMap count mismatch"); VarVec vec = this.CreateVarVec(); foreach (Var v in leftMap.Keys) { vec.Set(v); } return new UnionAllOp(vec, leftMap, rightMap, branchDiscriminator); } /// /// Creates a new IntersectOp /// /// Mappings from the Output Vars to the Vars produced by the left argument /// Mappings from the Output Vars to the Vars produced by the right argument /// An IntersectOp that references the specified left and right Vars internal IntersectOp CreateIntersectOp(VarMap leftMap, VarMap rightMap) { Debug.Assert(leftMap.Count == rightMap.Count, "VarMap count mismatch"); VarVec vec = this.CreateVarVec(); foreach (Var v in leftMap.Keys) { vec.Set(v); } return new IntersectOp(vec, leftMap, rightMap); } /// /// Creates a new ExceptOp /// /// Mappings from the Output Vars to the Vars produced by the left argument /// Mappings from the Output Vars to the Vars produced by the right argument /// An ExceptOp that references the specified left and right Vars internal ExceptOp CreateExceptOp(VarMap leftMap, VarMap rightMap) { Debug.Assert(leftMap.Count == rightMap.Count, "VarMap count mismatch"); VarVec vec = this.CreateVarVec(); foreach (Var v in leftMap.Keys) { vec.Set(v); } return new ExceptOp(vec, leftMap, rightMap); } /// /// Create a single-row-op (the relop analog of Element) /// /// internal SingleRowOp CreateSingleRowOp() { return SingleRowOp.Instance; } /// /// Create a SingleRowTableOp - a table with exactly one row (and no columns) /// /// internal SingleRowTableOp CreateSingleRowTableOp() { return SingleRowTableOp.Instance; } #endregion #region PhysicalOps /// /// Create a PhysicalProjectOp - with a columnMap describing the output /// /// list of output vars /// columnmap describing the output element /// internal PhysicalProjectOp CreatePhysicalProjectOp(VarList outputVars, SimpleCollectionColumnMap columnMap) { return new PhysicalProjectOp(outputVars, columnMap); } /// /// Create a physicalProjectOp - with a single column output /// /// the output element /// internal PhysicalProjectOp CreatePhysicalProjectOp(Var outputVar) { VarList varList = Command.CreateVarList(); varList.Add(outputVar); VarRefColumnMap varRefColumnMap = new VarRefColumnMap(outputVar); SimpleCollectionColumnMap collectionColumnMap = new SimpleCollectionColumnMap( TypeUtils.CreateCollectionType(varRefColumnMap.Type), // type null, // name varRefColumnMap, // element map new SimpleColumnMap[0], // keys new SimpleColumnMap[0]); // foreign keys return CreatePhysicalProjectOp(varList, collectionColumnMap); } /// /// Another overload - with an additional discriminatorValue. /// Should this be a subtype instead? /// /// the collectionVar /// column map for the collection element /// elementVars with any nested collections pulled up /// keys specific to this collection /// sort keys specific to this collecion /// discriminator value for this collection (under the current nestOp) /// a new CollectionInfo instance internal static CollectionInfo CreateCollectionInfo(Var collectionVar, ColumnMap columnMap, VarList flattenedElementVars, VarVec keys, List sortKeys, object discriminatorValue) { return new CollectionInfo(collectionVar, columnMap, flattenedElementVars, keys, sortKeys, discriminatorValue); } /// /// Create a singleStreamNestOp /// /// keys for the nest operation /// list of prefix sort keys /// list of postfix sort keys /// List of outputVars /// CollectionInfo for each collection /// Var describing the discriminator /// internal SingleStreamNestOp CreateSingleStreamNestOp(VarVec keys, List prefixSortKeys, List postfixSortKeys, VarVec outputVars, List collectionInfoList, Var discriminatorVar) { return new SingleStreamNestOp(keys, prefixSortKeys, postfixSortKeys, outputVars, collectionInfoList, discriminatorVar); } /// /// Create a MultiStreamNestOp /// /// list of prefix sort keys /// List of outputVars /// CollectionInfo for each collection element /// internal MultiStreamNestOp CreateMultiStreamNestOp(List prefixSortKeys, VarVec outputVars, List collectionInfoList) { return new MultiStreamNestOp(prefixSortKeys, outputVars, collectionInfoList); } #endregion #region NodeInfo /// /// Get auxilliary information for a Node /// /// the node /// node info for this node internal NodeInfo GetNodeInfo(Node n) { return n.GetNodeInfo(this); } /// /// Get extended node information for a RelOpNode /// /// the node /// extended node info for this node internal ExtendedNodeInfo GetExtendedNodeInfo(Node n) { return n.GetExtendedNodeInfo(this); } /// /// Recompute the nodeinfo for a node, but only if has already been computed /// /// Node in question internal void RecomputeNodeInfo(Node n) { m_nodeInfoVisitor.RecomputeNodeInfo(n); } #endregion #region KeyInfo /// /// Pulls up keys if necessary and gets the key information for a Node /// /// node /// key information internal KeyVec PullupKeys(Node n) { return m_keyPullupVisitor.GetKeys(n); } #endregion #region Type Comparisons // // The functions described in this region are used through out the // PlanCompiler to reason about type equality. Make sure that you // use these and these alone // /// /// Check to see if two types are considered "equal" for the purposes /// of the plan compiler. /// Two types are considered to be equal if their "identities" are equal. /// /// /// /// true, if the types are "equal" internal static bool EqualTypes(TypeUsage x, TypeUsage y) { return PlanCompiler.TypeUsageEqualityComparer.Instance.Equals(x, y); } /// /// Check to see if two types are considered "equal" for the purposes /// of the plan compiler /// /// /// /// true, if the types are "equal" internal static bool EqualTypes(EdmType x, EdmType y) { return PlanCompiler.TypeUsageEqualityComparer.Equals(x, y); } #endregion #region Builder Methods /// /// Builds out a UNION-ALL ladder from a sequence of node,var pairs. /// Assumption: Each node produces exactly one Var /// /// If the input sequence has zero elements, we return null /// If the input sequence has one element, we return that single element /// Otherwise, we build out a UnionAll ladder from each of the inputs. If the input sequence was {A,B,C,D}, /// we build up a union-all ladder that looks like /// (((A UA B) UA C) UA D) /// /// list of input nodes - one for each branch /// list of input vars - N for each branch /// the resulting union-all subtree /// the output vars from the union-all subtree internal void BuildUnionAllLadder( IList inputNodes, IList inputVars, out Node resultNode, out IList resultVars) { if (inputNodes.Count == 0) { resultNode = null; resultVars = null; return; } int varPerNode = inputVars.Count / inputNodes.Count; Debug.Assert((inputVars.Count % inputNodes.Count == 0) && (varPerNode >= 1), "Inconsistent nodes/vars count:" + inputNodes.Count + "," + inputVars.Count); if (inputNodes.Count == 1) { resultNode = inputNodes[0]; resultVars = inputVars; return; } List unionAllVars = new List(); Node unionAllNode = inputNodes[0]; for (int j = 0; j < varPerNode; j++) { unionAllVars.Add(inputVars[j]); } for (int i = 1; i < inputNodes.Count; i++) { VarMap leftVarMap = this.CreateVarMap(); VarMap rightVarMap = this.CreateVarMap(); List setOpVars = new List(); for (int j = 0; j < varPerNode; j++) { SetOpVar newVar = this.CreateSetOpVar(unionAllVars[j].Type); setOpVars.Add(newVar); leftVarMap.Add(newVar, unionAllVars[j]); rightVarMap.Add(newVar, inputVars[i * varPerNode + j]); } Op unionAllOp = this.CreateUnionAllOp(leftVarMap, rightVarMap); unionAllNode = this.CreateNode(unionAllOp, unionAllNode, inputNodes[i]); unionAllVars = setOpVars; } resultNode = unionAllNode; resultVars = unionAllVars; } /// /// A simplified version of the method above - each branch can produce only one var /// /// /// /// /// internal void BuildUnionAllLadder(IList inputNodes, IList inputVars, out Node resultNode, out Var resultVar) { Debug.Assert(inputNodes.Count == inputVars.Count, "Count mismatch:" + inputNodes.Count + "," + inputVars.Count); IList varList; BuildUnionAllLadder(inputNodes, inputVars, out resultNode, out varList); if (varList != null && varList.Count > 0) { resultVar = varList[0]; } else { resultVar = null; } } /// /// Build a projectOp tree over the input. /// This function builds a projectOp tree over the input. The Outputs (vars) of the project are the /// list of vars from the input (inputVars), plus one computed Var for each of the computed expressions /// (computedExpressions) /// /// the input relop to the project /// List of vars from the input that need to be projected /// list (possibly empty) of any computed expressions /// internal Node BuildProject(Node inputNode, IEnumerable inputVars, IEnumerable computedExpressions) { Debug.Assert(inputNode.Op.IsRelOp, "Expected a RelOp. Found " + inputNode.Op.OpType); VarDefListOp varDefListOp = this.CreateVarDefListOp(); Node varDefListNode = this.CreateNode(varDefListOp); VarVec projectVars = this.CreateVarVec(inputVars); foreach (Node expr in computedExpressions) { Var v = this.CreateComputedVar(expr.Op.Type); projectVars.Set(v); VarDefOp varDefOp = this.CreateVarDefOp(v); Node varDefNode = this.CreateNode(varDefOp, expr); varDefListNode.Children.Add(varDefNode); } Node projectNode = this.CreateNode( this.CreateProjectOp(projectVars), inputNode, varDefListNode); return projectNode; } /// /// A "simpler" builder method for ProjectOp. The assumption is that the only output is the /// (var corresponding to) the computedExpression. None of the Vars of the "input" are projected out /// /// The single output Var is returned in the "outputVar" parameter /// /// the input relop /// the computed expression /// (output) the computed var corresponding to the computed expression /// the new project subtree node internal Node BuildProject(Node input, Node computedExpression, out Var projectVar) { Node projectNode = BuildProject(input, new Var[] { }, new Node[] { computedExpression }); projectVar = ((ProjectOp)projectNode.Op).Outputs.First; return projectNode; } /// /// Build the equivalent of an OfTypeExpression over the input (ie) produce the set of values from the /// input that are of the desired type (exactly of the desired type, if the "includeSubtypes" parameter is false). /// /// Further more, "update" the result element type to be the desired type. /// /// We accomplish this by first building a FilterOp with an IsOf (or an IsOfOnly) predicate for the desired /// type. We then build out a ProjectOp over the FilterOp, where we introduce a "Fake" TreatOp over the input /// element to cast it to the right type. The "Fake" TreatOp is only there for "compile-time" typing reasons, /// and will be ignored in the rest of the plan compiler /// /// the input collection /// the single Var produced by the input collection /// the desired element type /// do we include subtypes of the desired element type /// the result subtree /// the single Var produced by the result subtree internal void BuildOfTypeTree(Node inputNode, Var inputVar, TypeUsage desiredType, bool includeSubtypes, out Node resultNode, out Var resultVar) { Op isOfOp = includeSubtypes ? this.CreateIsOfOp(desiredType) : this.CreateIsOfOnlyOp(desiredType); Node predicate = this.CreateNode(isOfOp, this.CreateNode(this.CreateVarRefOp(inputVar))); Node filterNode = this.CreateNode(this.CreateFilterOp(), inputNode, predicate); resultNode = BuildFakeTreatProject(filterNode, inputVar, desiredType, out resultVar); } /// Builds out a ProjectOp over the input that introduces a "Fake" TreatOp over the input Var to cast it to the desired type /// The "Fake" TreatOp is only there for "compile-time" typing reasons, and will be ignored in the rest of the plan compiler. /// /// the input collection /// the single Var produced by the input collection /// the desired element type /// the single Var produced by the result subtree /// the result subtree internal Node BuildFakeTreatProject(Node inputNode, Var inputVar, TypeUsage desiredType, out Var resultVar) { Node treatNode = this.CreateNode(this.CreateFakeTreatOp(desiredType), this.CreateNode(this.CreateVarRefOp(inputVar))); Node resultNode = this.BuildProject(inputNode, treatNode, out resultVar); return resultNode; } /// /// Build a comparisonOp over the input arguments. Build SoftCasts over the inputs, if we need /// to. /// /// the comparison optype /// Arg 0 /// Arg 1 /// the resulting comparison tree internal Node BuildComparison(OpType opType, Node arg0, Node arg1) { if (!Command.EqualTypes(arg0.Op.Type, arg1.Op.Type)) { TypeUsage commonType = TypeHelpers.GetCommonTypeUsage(arg0.Op.Type, arg1.Op.Type); Debug.Assert(commonType != null, "No common type for " + arg0.Op.Type + " and " + arg1.Op.Type); if (!EqualTypes(commonType, arg0.Op.Type)) { arg0 = this.CreateNode(this.CreateSoftCastOp(commonType), arg0); } if (!EqualTypes(commonType, arg1.Op.Type)) { arg1 = this.CreateNode(this.CreateSoftCastOp(commonType), arg1); } } Node newNode = this.CreateNode(this.CreateComparisonOp(opType), arg0, arg1); return newNode; } /// /// Build up a CollectOp over a relop tree /// /// the relop tree /// the single output var from the relop tree /// internal Node BuildCollect(Node relOpNode, Var relOpVar) { Node physicalProjectNode = this.CreateNode(this.CreatePhysicalProjectOp(relOpVar), relOpNode); TypeUsage collectOpType = TypeHelpers.CreateCollectionTypeUsage(relOpVar.Type); Node collectNode = this.CreateNode(this.CreateCollectOp(collectOpType), physicalProjectNode); return collectNode; } #endregion #region Rel Properties /// /// Mark this rel-property as "referenced" in the current query, if the target /// end has multiplicity of one (or zero_or_one) /// /// the rel-property private void AddRelPropertyReference(RelProperty relProperty) { if (relProperty.ToEnd.RelationshipMultiplicity != RelationshipMultiplicity.Many && !m_referencedRelProperties.Contains(relProperty)) { m_referencedRelProperties.Add(relProperty); } } /// /// The set of referenced rel properties in the current query /// internal HashSet ReferencedRelProperties { get { return m_referencedRelProperties; } } /// /// Is this rel-property referenced in the query so far /// /// the rel-property /// true, if the rel property was referenced in the query internal bool IsRelPropertyReferenced(RelProperty relProperty) { bool ret = m_referencedRelProperties.Contains(relProperty); return ret; } #endregion #endregion } }