You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			566 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			566 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //---------------------------------------------------------------------
 | |
| // <copyright file="PlanCompiler.cs" company="Microsoft">
 | |
| //      Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>
 | |
| //
 | |
| // @owner  Microsoft
 | |
| // @backupOwner Microsoft
 | |
| //---------------------------------------------------------------------
 | |
| 
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Diagnostics; // Please use PlanCompiler.Assert instead of Debug.Assert in this class...
 | |
| 
 | |
| // It is fine to use Debug.Assert in cases where you assert an obvious thing that is supposed
 | |
| // to prevent from simple mistakes during development (e.g. method argument validation 
 | |
| // in cases where it was you who created the variables or the variables had already been validated or 
 | |
| // in "else" clauses where due to code changes (e.g. adding a new value to an enum type) the default 
 | |
| // "else" block is chosen why the new condition should be treated separately). This kind of asserts are 
 | |
| // (can be) helpful when developing new code to avoid simple mistakes but have no or little value in 
 | |
| // the shipped product. 
 | |
| // PlanCompiler.Assert *MUST* be used to verify conditions in the trees. These would be assumptions 
 | |
| // about how the tree was built etc. - in these cases we probably want to throw an exception (this is
 | |
| // what PlanCompiler.Assert does when the condition is not met) if either the assumption is not correct 
 | |
| // or the tree was built/rewritten not the way we thought it was.
 | |
| // Use your judgment - if you rather remove an assert than ship it use Debug.Assert otherwise use
 | |
| // PlanCompiler.Assert.
 | |
| 
 | |
| using cqt = System.Data.Common.CommandTrees;
 | |
| using md = System.Data.Metadata.Edm;
 | |
| using System.Data.Query.InternalTrees;
 | |
| using System.Data.Query.PlanCompiler;
 | |
| 
 | |
| namespace System.Data.Query.PlanCompiler
 | |
| {
 | |
|     /// <summary>
 | |
|     /// The PlanCompiler class is used by the BridgeCommand to produce an 
 | |
|     /// execution plan - this execution plan is the plan object. The plan compilation
 | |
|     /// process takes as input a command tree (in C space), and then runs through a 
 | |
|     /// set of changes before the final plan is produced. The final plan contains 
 | |
|     /// one or more command trees (commands?) (in S space), with a set of assembly
 | |
|     /// instructions.
 | |
|     /// The compiler phases include
 | |
|     /// * Convert the command tree (CTree) into an internal tree (an ITree)
 | |
|     /// * Run initializations on the ITree. 
 | |
|     /// * Eliminate structured types from the tree
 | |
|     ///    * Eliminating named type references, refs and records from the tree
 | |
|     ///    At the end of this phase, we still may have collections (and record 
 | |
|     ///    arguments to collections) in the tree.
 | |
|     /// * Projection pruning (ie) eliminating unused references
 | |
|     /// * Tree transformations. Various transformations are run on the ITree to
 | |
|     ///      (ostensibly) optimize the tree. These transformations are represented as
 | |
|     ///      rules, and a rule processor is invoked.
 | |
|     /// * Nest elimination. At this point, we try to get pull up nest operations
 | |
|     ///      as high up the tree as possible
 | |
|     /// * Code Generation. This phase produces a plan object with various subpieces 
 | |
|     ///      of the ITree represented as commands (in S space).
 | |
|     ///    * The subtrees of the ITree are then converted into the corresponding CTrees
 | |
|     ///      and converted into S space as part of the CTree creation.
 | |
|     ///    * A plan object is created and returned.
 | |
|     /// </summary>
 | |
|     internal class PlanCompiler
 | |
|     {
 | |
| 
 | |
|         #region private state
 | |
| 
 | |
|         /// <summary>
 | |
|         /// A boolean switch indicating whether we should apply transformation rules regardless of the size of the Iqt.
 | |
|         /// By default, the Enabled property of a boolean switch is set using the value specified in the configuration file. 
 | |
|         /// Configuring the switch with a value of 0 sets the Enabled property to false; configuring the switch with a nonzero 
 | |
|         /// value to set the Enabled property to true. If the BooleanSwitch constructor cannot find initial switch settings 
 | |
|         /// in the configuration file, the Enabled property of the new switch is set to false by default.
 | |
|         /// </summary>
 | |
|         private static BooleanSwitch _applyTransformationsRegardlessOfSize = new BooleanSwitch("System.Data.EntityClient.IgnoreOptimizationLimit", "The Entity Framework should try to optimize the query regardless of its size");
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Determines the maximum size of the query in terms of Iqt nodes for which we attempt to do transformation rules.
 | |
|         /// This number is ignored if applyTransformationsRegardlessOfSize is enabled.
 | |
|         /// </summary>
 | |
|         private const int MaxNodeCountForTransformations = 100000;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The CTree we're compiling a plan for.
 | |
|         /// </summary>
 | |
|         private cqt.DbCommandTree m_ctree;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The ITree we're working on.
 | |
|         /// </summary>
 | |
|         private Command m_command;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The phase of the process we're currently in.
 | |
|         /// </summary>
 | |
|         private PlanCompilerPhase m_phase;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Set of phases we need to go through
 | |
|         /// </summary>
 | |
|         private int m_neededPhases;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Keeps track of foreign key relationships. Needed by Join Elimination
 | |
|         /// </summary>
 | |
|         private ConstraintManager m_constraintManager;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Can transformation rules be applied
 | |
|         /// </summary>
 | |
|         private Nullable<bool> m_mayApplyTransformationRules = null;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Does the command include any sort key that represents a null sentinel
 | |
|         /// This may only be set to true in NominalTypeElimination and is used
 | |
|         /// in Transformation Rules
 | |
|         /// </summary>
 | |
|         private bool m_hasSortingOnNullSentinels = false;
 | |
|         #endregion
 | |
| 
 | |
|         #region constructors
 | |
| 
 | |
|         /// <summary>
 | |
|         /// private constructor
 | |
|         /// </summary>
 | |
|         /// <param name="ctree">the input cqt</param>
 | |
|         private PlanCompiler(cqt.DbCommandTree ctree)
 | |
|         {
 | |
|             m_ctree = ctree; // the input command tree
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
| 
 | |
|         #region public interfaces
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retail Assertion code.
 | |
|         /// 
 | |
|         /// Provides the ability to have retail asserts.
 | |
|         /// </summary>
 | |
|         /// <param name="condition"></param>
 | |
|         /// <param name="message"></param>
 | |
|         internal static void Assert(bool condition, string message)
 | |
|         {
 | |
|             if (!condition)
 | |
|             {
 | |
|                 System.Diagnostics.Debug.Fail(message);
 | |
| 
 | |
|                 // NOTE: I considered, at great length, whether to have the assertion message text 
 | |
|                 //       included in the exception we throw; in the end, there really isn't a reliable
 | |
|                 //       equivalent to the C++ __LINE__ and __FILE__ macros in C# (at least not without
 | |
|                 //       using the C++ PreProcessor...ick)  The StackTrace object comes close but 
 | |
|                 //       doesn't handle inlined callers properly for our needs (MethodA() calls MethodB() 
 | |
|                 //       calls us, but MethodB() is inlined, so we'll get MethodA() info instead), and
 | |
|                 //       since these are retail "Asserts" (as in: we're not supposed to get them in our 
 | |
|                 //       shipping code, and we're doing this to avoid a null-ref which is even worse) I
 | |
|                 //       elected to simplify this by just including them as the additional info.
 | |
|                 throw EntityUtil.InternalError(EntityUtil.InternalErrorCode.AssertionFailed, 0, message);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compile a query, and produce a plan
 | |
|         /// </summary>
 | |
|         /// <param name="ctree">the input CQT</param>
 | |
|         /// <param name="providerCommands">list of provider commands</param>
 | |
|         /// <param name="resultColumnMap">column map for result assembly</param>
 | |
|         /// <param name="entitySets">the entity sets referenced in this query</param>
 | |
|         /// <returns>a compiled plan object</returns>
 | |
|         internal static void Compile(cqt.DbCommandTree ctree, out List<ProviderCommandInfo> providerCommands, out ColumnMap resultColumnMap, out int columnCount, out Common.Utils.Set<md.EntitySet> entitySets)
 | |
|         {
 | |
|             PlanCompiler.Assert(ctree != null, "Expected a valid, non-null Command Tree input");
 | |
|             PlanCompiler pc = new PlanCompiler(ctree);
 | |
|             pc.Compile(out providerCommands, out resultColumnMap, out columnCount, out entitySets);
 | |
|         }
 | |
| 
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Get the current command
 | |
|         /// </summary>
 | |
|         internal Command Command { get { return m_command; } }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Does the command include any sort key that represents a null sentinel
 | |
|         /// This may only be set to true in NominalTypeElimination and is used
 | |
|         /// in Transformation Rules
 | |
|         /// </summary>
 | |
|         internal bool HasSortingOnNullSentinels 
 | |
|         {
 | |
|             get { return m_hasSortingOnNullSentinels; } 
 | |
|             set { m_hasSortingOnNullSentinels = value; }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Keeps track of foreign key relationships. Needed by  Join Elimination
 | |
|         /// </summary>
 | |
|         internal ConstraintManager ConstraintManager
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (m_constraintManager == null)
 | |
|                 {
 | |
|                     m_constraintManager = new ConstraintManager();
 | |
|                 }
 | |
|                 return m_constraintManager;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if DEBUG
 | |
|         /// <summary>
 | |
|         /// Get the current plan compiler phase
 | |
|         /// </summary>
 | |
|         internal PlanCompilerPhase Phase { get { return m_phase; } }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Sets the current plan compiler trace function to <paramref name="traceCallback"/>, enabling plan compiler tracing
 | |
|         /// </summary>
 | |
|         internal static void TraceOn(Action<string, object> traceCallback)
 | |
|         {
 | |
|             s_traceCallback = traceCallback;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Sets the current plan compiler trace function to <c>null</c>, disabling plan compiler tracing
 | |
|         /// </summary>
 | |
|         internal static void TraceOff()
 | |
|         {
 | |
|             s_traceCallback = null;
 | |
|         }
 | |
| 
 | |
|         private static Action<string, object> s_traceCallback;
 | |
| #endif
 | |
|         /// <summary>
 | |
|         /// The MetadataWorkspace
 | |
|         /// </summary>
 | |
|         internal md.MetadataWorkspace MetadataWorkspace { get { return m_ctree.MetadataWorkspace; } }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Is the specified phase needed for this query?
 | |
|         /// </summary>
 | |
|         /// <param name="phase">the phase in question</param>
 | |
|         /// <returns></returns>
 | |
|         internal bool IsPhaseNeeded(PlanCompilerPhase phase)
 | |
|         {
 | |
|             return ((m_neededPhases & (1 << (int)phase)) != 0);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Mark the specified phase as needed
 | |
|         /// </summary>
 | |
|         /// <param name="phase">plan compiler phase</param>
 | |
|         internal void MarkPhaseAsNeeded(PlanCompilerPhase phase)
 | |
|         {
 | |
|             m_neededPhases = m_neededPhases | (1 << (int)phase);
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
| 
 | |
|         #region private methods
 | |
| 
 | |
|         /// <summary>
 | |
|         /// The real driver. 
 | |
|         /// </summary>
 | |
|         /// <param name="providerCommands">list of provider commands</param>
 | |
|         /// <param name="resultColumnMap">column map for the result</param>
 | |
|         /// <param name="entitySets">the entity sets exposed in this query</param>
 | |
|         private void Compile(out List<ProviderCommandInfo> providerCommands, out ColumnMap resultColumnMap, out int columnCount, out Common.Utils.Set<md.EntitySet> entitySets)
 | |
|         {
 | |
|             Initialize(); // initialize the ITree
 | |
| 
 | |
|             string beforePreProcessor = String.Empty;
 | |
|             string beforeAggregatePushdown = String.Empty;
 | |
|             string beforeNormalization = String.Empty;
 | |
|             string beforeNTE = String.Empty;
 | |
|             string beforeProjectionPruning1 = String.Empty;
 | |
|             string beforeNestPullup = String.Empty;
 | |
|             string beforeProjectionPruning2 = String.Empty;
 | |
|             string beforeTransformationRules1 = String.Empty;
 | |
|             string beforeProjectionPruning3 = String.Empty;
 | |
|             string beforeTransformationRules2 = String.Empty;
 | |
|             string beforeJoinElimination1 = String.Empty;
 | |
|             string beforeTransformationRules3 = String.Empty;
 | |
|             string beforeJoinElimination2 = String.Empty;
 | |
|             string beforeTransformationRules4 = String.Empty;
 | |
|             string beforeCodeGen = String.Empty;
 | |
| 
 | |
|             //
 | |
|             // We always need the pre-processor and the codegen phases.
 | |
|             // It is generally a good thing to run through the transformation rules, and 
 | |
|             // the projection pruning phases.
 | |
|             // The "optional" phases are AggregatePushdown, Normalization, NTE, NestPullup and JoinElimination
 | |
|             //
 | |
|             m_neededPhases = (1 << (int)PlanCompilerPhase.PreProcessor) |
 | |
|                 // (1 << (int)PlanCompilerPhase.AggregatePushdown) |
 | |
|                 // (1 << (int)PlanCompilerPhase.Normalization) |
 | |
|                 // (1 << (int)PlanCompilerPhase.NTE) |
 | |
|                 (1 << (int)PlanCompilerPhase.ProjectionPruning) |
 | |
|                 // (1 << (int)PlanCompilerPhase.NestPullup) |
 | |
|                 (1 << (int)PlanCompilerPhase.Transformations) |
 | |
|                 // (1 << (int)PlanCompilerPhase.JoinElimination) |
 | |
|                 (1 << (int)PlanCompilerPhase.CodeGen);
 | |
| 
 | |
|             // Perform any necessary preprocessing
 | |
|             StructuredTypeInfo typeInfo;
 | |
|             Dictionary<md.EdmFunction, md.EdmProperty[]> tvfResultKeys;
 | |
|             beforePreProcessor = SwitchToPhase(PlanCompilerPhase.PreProcessor);
 | |
|             PreProcessor.Process(this, out typeInfo, out tvfResultKeys);
 | |
|             entitySets = typeInfo.GetEntitySets();
 | |
| 
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.AggregatePushdown))
 | |
|             {
 | |
|                 beforeAggregatePushdown = SwitchToPhase(PlanCompilerPhase.AggregatePushdown);
 | |
|                 AggregatePushdown.Process(this);
 | |
|             }
 | |
| 
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.Normalization))
 | |
|             {
 | |
|                 beforeNormalization = SwitchToPhase(PlanCompilerPhase.Normalization);
 | |
|                 Normalizer.Process(this);
 | |
|             }
 | |
| 
 | |
|             // Eliminate "structured" types.
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.NTE))
 | |
|             {
 | |
|                 beforeNTE = SwitchToPhase(PlanCompilerPhase.NTE);
 | |
|                 NominalTypeEliminator.Process(this, typeInfo, tvfResultKeys);
 | |
|             }
 | |
| 
 | |
|             // Projection pruning - eliminate unreferenced expressions
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.ProjectionPruning))
 | |
|             {
 | |
|                 beforeProjectionPruning1 = SwitchToPhase(PlanCompilerPhase.ProjectionPruning);
 | |
|                 ProjectionPruner.Process(this);
 | |
|             }
 | |
| 
 | |
|             // Nest Pull-up on the ITree
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.NestPullup))
 | |
|             {
 | |
|                 beforeNestPullup = SwitchToPhase(PlanCompilerPhase.NestPullup);
 | |
| 
 | |
|                 NestPullup.Process(this);
 | |
| 
 | |
|                 //If we do Nest Pull-up, we should again do projection pruning
 | |
|                 beforeProjectionPruning2 = SwitchToPhase(PlanCompilerPhase.ProjectionPruning);
 | |
|                 ProjectionPruner.Process(this);
 | |
|             }
 | |
| 
 | |
|             // Run transformations on the tree
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.Transformations))
 | |
|             {
 | |
|                 bool projectionPrunningNeeded = ApplyTransformations(ref beforeTransformationRules1, TransformationRulesGroup.All);
 | |
| 
 | |
|                 if (projectionPrunningNeeded)
 | |
|                 {
 | |
|                     beforeProjectionPruning3 = SwitchToPhase(PlanCompilerPhase.ProjectionPruning);
 | |
|                     ProjectionPruner.Process(this);
 | |
|                     ApplyTransformations(ref beforeTransformationRules2, TransformationRulesGroup.Project);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Join elimination
 | |
|             if (IsPhaseNeeded(PlanCompilerPhase.JoinElimination))
 | |
|             {
 | |
|                 beforeJoinElimination1 = SwitchToPhase(PlanCompilerPhase.JoinElimination);
 | |
|                 bool modified = JoinElimination.Process(this);
 | |
|                 if (modified)
 | |
|                 {
 | |
|                     ApplyTransformations(ref beforeTransformationRules3, TransformationRulesGroup.PostJoinElimination);
 | |
|                     beforeJoinElimination2 = SwitchToPhase(PlanCompilerPhase.JoinElimination);
 | |
|                     modified = JoinElimination.Process(this);
 | |
|                     if (modified)
 | |
|                     {
 | |
|                         ApplyTransformations(ref beforeTransformationRules4, TransformationRulesGroup.PostJoinElimination);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Code generation
 | |
|             beforeCodeGen = SwitchToPhase(PlanCompilerPhase.CodeGen);
 | |
|             CodeGen.Process(this, out providerCommands, out resultColumnMap, out columnCount);
 | |
| 
 | |
| #if DEBUG
 | |
|             // GC.KeepAlive makes FxCop Grumpy.
 | |
|             int size = beforePreProcessor.Length;
 | |
|             size = beforeAggregatePushdown.Length;
 | |
|             size = beforeNormalization.Length;
 | |
|             size = beforeNTE.Length;
 | |
|             size = beforeProjectionPruning1.Length;
 | |
|             size = beforeNestPullup.Length;
 | |
|             size = beforeProjectionPruning2.Length;
 | |
|             size = beforeTransformationRules1.Length;
 | |
|             size = beforeProjectionPruning3.Length;
 | |
|             size = beforeTransformationRules2.Length;
 | |
|             size = beforeJoinElimination1.Length;
 | |
|             size = beforeTransformationRules3.Length;
 | |
|             size = beforeJoinElimination2.Length;
 | |
|             size = beforeTransformationRules4.Length;
 | |
|             size = beforeCodeGen.Length;
 | |
| #endif
 | |
|             // All done
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Helper method for applying transformation rules
 | |
|         /// </summary>
 | |
|         /// <param name="dumpString"></param>
 | |
|         /// <param name="rulesGroup"></param>
 | |
|         /// <returns></returns>
 | |
|         private bool ApplyTransformations(ref string dumpString, TransformationRulesGroup rulesGroup)
 | |
|         {
 | |
|             if (MayApplyTransformationRules)
 | |
|             {
 | |
|                 dumpString = SwitchToPhase(PlanCompilerPhase.Transformations);
 | |
|                 return TransformationRules.Process(this, rulesGroup);
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Logic to perform between each compile phase
 | |
|         /// </summary>
 | |
|         /// <param name="newPhase"></param>
 | |
|         /// <returns></returns>
 | |
|         private string SwitchToPhase(PlanCompilerPhase newPhase)
 | |
|         {
 | |
|             string iqtDumpResult = string.Empty;
 | |
| 
 | |
|             m_phase = newPhase;
 | |
| 
 | |
| #if DEBUG
 | |
|             if (s_traceCallback != null)
 | |
|             {
 | |
|                 s_traceCallback(Enum.GetName(typeof(PlanCompilerPhase), newPhase), m_command);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 iqtDumpResult = Dump.ToXml(m_command);
 | |
|             }
 | |
| 
 | |
|             Validator.Validate(this);
 | |
| #endif
 | |
|             return iqtDumpResult;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// To avoid processing huge trees, transformation rules are applied only if the number of nodes 
 | |
|         /// is less than MaxNodeCountForTransformations
 | |
|         /// or if it is specified that they should be applied regardless of the size of the query.
 | |
|         /// Whether to apply transformations is only computed the first time this property is requested, 
 | |
|         /// and is cached afterwards. This is because we don't expect the tree to get larger 
 | |
|         /// from applying transformations. 
 | |
|         /// </summary>
 | |
|         private bool MayApplyTransformationRules
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (m_mayApplyTransformationRules == null)
 | |
|                 {
 | |
|                     m_mayApplyTransformationRules = ComputeMayApplyTransformations();
 | |
|                 }
 | |
|                 return m_mayApplyTransformationRules.Value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Compute whether transformations may be applied. 
 | |
|         /// Transformation rules may be applied only if the number of nodes is less than 
 | |
|         /// MaxNodeCountForTransformations or if it is specified that they should be applied 
 | |
|         /// regardless of the size of the query.
 | |
|         /// </summary>
 | |
|         /// <returns></returns>
 | |
|         private bool ComputeMayApplyTransformations()
 | |
|         {
 | |
|             //
 | |
|             // If the nextNodeId is less than MaxNodeCountForTransformations then we don't need to 
 | |
|             // calculate the acutal node count, it must be less than  MaxNodeCountForTransformations
 | |
|             //
 | |
|             if (_applyTransformationsRegardlessOfSize.Enabled || this.m_command.NextNodeId < MaxNodeCountForTransformations)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             //Compute the actual node count
 | |
|             int actualCount = NodeCounter.Count(this.m_command.Root);
 | |
|             return (actualCount < MaxNodeCountForTransformations);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Converts the CTree into an ITree, and initializes the plan
 | |
|         /// </summary>
 | |
|         private void Initialize()
 | |
|         {
 | |
|             // Only support queries for now
 | |
|             cqt.DbQueryCommandTree cqtree = m_ctree as cqt.DbQueryCommandTree;
 | |
|             PlanCompiler.Assert(cqtree != null, "Unexpected command tree kind. Only query command tree is supported.");
 | |
| 
 | |
|             // Generate the ITree
 | |
|             m_command = ITreeGenerator.Generate(cqtree);
 | |
|             PlanCompiler.Assert(m_command != null, "Unable to generate internal tree from Command Tree");
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
|     }
 | |
| 
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Enum describing which phase of plan compilation we're currently in
 | |
|     /// </summary>
 | |
|     internal enum PlanCompilerPhase
 | |
|     {
 | |
|         /// <summary>
 | |
|         /// Just entering the PreProcessor phase
 | |
|         /// </summary>
 | |
|         PreProcessor = 0,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the AggregatePushdown phase
 | |
|         /// </summary>
 | |
|         AggregatePushdown = 1,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the Normalization phase
 | |
|         /// </summary>
 | |
|         Normalization = 2,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the NTE (Nominal Type Eliminator) phase
 | |
|         /// </summary>
 | |
|         NTE = 3,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the Projection pruning phase
 | |
|         /// </summary>
 | |
|         ProjectionPruning = 4,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the Nest Pullup phase
 | |
|         /// </summary>
 | |
|         NestPullup = 5,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the Transformations phase
 | |
|         /// </summary>
 | |
|         Transformations = 6,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the JoinElimination phase
 | |
|         /// </summary>
 | |
|         JoinElimination = 7,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Entering the codegen phase
 | |
|         /// </summary>
 | |
|         CodeGen = 8,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// We're almost done
 | |
|         /// </summary>
 | |
|         PostCodeGen = 9,
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Marker
 | |
|         /// </summary>
 | |
|         MaxMarker = 10
 | |
|     }
 | |
| }
 |