You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			628 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			628 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //---------------------------------------------------------------------
 | |
| // <copyright file="CellTreeSimplifier.cs" company="Microsoft">
 | |
| //      Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>
 | |
| //
 | |
| // @owner [....]
 | |
| // @backupOwner [....]
 | |
| //---------------------------------------------------------------------
 | |
| 
 | |
| using System.Data.Common.Utils;
 | |
| using System.Data.Mapping.ViewGeneration.Structures;
 | |
| using System.Collections.Generic;
 | |
| using System.Text;
 | |
| using System.Diagnostics;
 | |
| using System.Data.Metadata.Edm;
 | |
| using System.Linq;
 | |
| 
 | |
| namespace System.Data.Mapping.ViewGeneration
 | |
| {
 | |
| 
 | |
|     // This class simplifies an extent's view. Given a view, runs the TM/SP
 | |
|     // rules to remove unnecessary self-joins or self-unions
 | |
|     internal class CellTreeSimplifier : InternalBase
 | |
|     {
 | |
| 
 | |
|         #region Fields
 | |
|         private ViewgenContext m_viewgenContext;
 | |
|         #endregion
 | |
| 
 | |
| 
 | |
|         #region Constructor
 | |
|         private CellTreeSimplifier(ViewgenContext context)
 | |
|         {
 | |
|             m_viewgenContext = context;
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         #region Exposed Methods
 | |
|         // effects: see CellTreeNode.Simplify below
 | |
|         internal static CellTreeNode MergeNodes(CellTreeNode rootNode)
 | |
|         {
 | |
|             CellTreeSimplifier simplifier = new CellTreeSimplifier(rootNode.ViewgenContext);
 | |
|             return simplifier.SimplifyTreeByMergingNodes(rootNode);
 | |
|         }
 | |
| 
 | |
|         // effects: Simplifies the tree rooted at rootNode and returns a new
 | |
|         // tree -- it ensures that the returned tree has at most one node for
 | |
|         // any particular extent unless the tree has nodes of the same extent
 | |
|         // embedded two leaves below LASJ or LOJ, e.g., if we have a tree
 | |
|         // (where Ni indicates a node for extent i - one Ni can be different
 | |
|         // from anohter Ni: 
 | |
|         // [N0 IJ N1] LASJ N0 --> This will not be simplified
 | |
|         // canBooleansOverlap indicates whether an original input cell
 | |
|         // contributes to multiple nodes in this tree, e.g., V1 IJ V2 UNION V2 IJ V3
 | |
|         private CellTreeNode SimplifyTreeByMergingNodes(CellTreeNode rootNode)
 | |
|         {
 | |
| 
 | |
|             if (rootNode is LeafCellTreeNode)
 | |
|             { // View already simple!
 | |
|                 return rootNode;
 | |
|             }
 | |
|             Debug.Assert(rootNode.OpType == CellTreeOpType.LOJ || rootNode.OpType == CellTreeOpType.IJ ||
 | |
|                          rootNode.OpType == CellTreeOpType.FOJ || rootNode.OpType == CellTreeOpType.Union ||
 | |
|                          rootNode.OpType == CellTreeOpType.LASJ,
 | |
|                          "Only handle these operations");
 | |
| 
 | |
|             // Before we apply any rule, check if we can improve the opportunity to
 | |
|             // collapse the nodes
 | |
|             rootNode = RestructureTreeForMerges(rootNode);
 | |
| 
 | |
|             List<CellTreeNode> children = rootNode.Children;
 | |
|             Debug.Assert(children.Count > 0, "OpCellTreeNode has no children?");
 | |
| 
 | |
|             // Apply recursively
 | |
|             for (int i = 0; i < children.Count; i++)
 | |
|             {
 | |
|                 children[i] = SimplifyTreeByMergingNodes(children[i]);
 | |
|             }
 | |
| 
 | |
|             // Essentially, we have a node with IJ, LOJ, U or FOJ type that
 | |
|             // has some children. Check if some of the children can be merged
 | |
|             // with one another using the corresponding TM/SP rule
 | |
| 
 | |
|             // Ops such as IJ, Union and FOJ are associative, i.e., A op (B
 | |
|             // op C) is the same as (A op B) op C. This is not true for LOJ
 | |
|             // and LASJ
 | |
|             bool isAssociativeOp = CellTreeNode.IsAssociativeOp(rootNode.OpType);
 | |
|             if (isAssociativeOp)
 | |
|             {
 | |
|                 // Group all the leaf cells of an extent together so that we can
 | |
|                 // later simply run through them without running nested loops
 | |
|                 // We do not do this for LOJ/LASJ nodes since LOJ (LASJ) is not commutative
 | |
|                 // (or associative);
 | |
|                 children = GroupLeafChildrenByExtent(children);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 children = GroupNonAssociativeLeafChildren(children);
 | |
|             }
 | |
| 
 | |
|             // childrenSet keeps track of the children that need to be procesed/partitioned
 | |
|             OpCellTreeNode newNode = new OpCellTreeNode(m_viewgenContext, rootNode.OpType);
 | |
|             CellTreeNode lastChild = null;
 | |
|             bool skipRest = false;
 | |
|             foreach (CellTreeNode child in children)
 | |
|             {
 | |
|                 if (lastChild == null)
 | |
|                 {
 | |
|                     // First time in the loop. Just set lastChild
 | |
|                     lastChild = child;
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 bool mergedOk = false;
 | |
|                 // try to merge lastChild and child
 | |
|                 if (false == skipRest && lastChild.OpType == CellTreeOpType.Leaf &&
 | |
|                     child.OpType == CellTreeOpType.Leaf)
 | |
|                 {
 | |
|                     // Both are cell queries. Can try to merge them
 | |
|                     // We do not add lastChild since it could merge
 | |
|                     // further. It will be added in a later loop or outside the loop
 | |
|                     mergedOk = TryMergeCellQueries(rootNode.OpType, ref lastChild, child);
 | |
|                 }
 | |
| 
 | |
|                 if (false == mergedOk)
 | |
|                 {
 | |
|                     // No merge occurred. Simply add the previous child as it
 | |
|                     // is (Note lastChild will be added in the next loop or if
 | |
|                     // the loop finishes, outside the loop
 | |
|                     newNode.Add(lastChild);
 | |
|                     lastChild = child;
 | |
|                     if (false == isAssociativeOp)
 | |
|                     {
 | |
|                         // LOJ is not associative:
 | |
|                         // (P loj PA) loj PO != P loj (PA loj PO). The RHS does not have
 | |
|                         // Persons who have orders but no addresses
 | |
|                         skipRest = true;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             newNode.Add(lastChild);
 | |
|             CellTreeNode result = newNode.AssociativeFlatten();
 | |
|             return result;
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         #region Private Methods
 | |
|         // effects: Restructure tree so that it is better positioned for merges
 | |
|         private CellTreeNode RestructureTreeForMerges(CellTreeNode rootNode)
 | |
|         {
 | |
|             List<CellTreeNode> children = rootNode.Children;
 | |
|             if (CellTreeNode.IsAssociativeOp(rootNode.OpType) == false || children.Count <= 1)
 | |
|             {
 | |
|                 return rootNode;
 | |
|             }
 | |
| 
 | |
|             // If this node's operator is associative and each child's
 | |
|             // operator is also associative, check if there is a common set
 | |
|             // of leaf nodes across all grandchildren
 | |
| 
 | |
|             Set<LeafCellTreeNode> commonGrandChildren = GetCommonGrandChildren(children);
 | |
|             if (commonGrandChildren == null)
 | |
|             {
 | |
|                 return rootNode;
 | |
|             }
 | |
| 
 | |
|             CellTreeOpType commonChildOpType = children[0].OpType;
 | |
| 
 | |
|             //  We do have the structure that we are looking for
 | |
|             // (common op2 gc2) op1 (common op2 gc3) op1 (common op2 gc4) becomes
 | |
|             // common op2 (gc2 op1 gc3 op1 gc4)
 | |
|             // e.g., (A IJ B IJ X IJ Y) UNION (A IJ B IJ Y IJ Z) UNION (A IJ B IJ R IJ S)
 | |
|             // becomes A IJ B IJ ((X IJ Y) UNION (Y IJ Z) UNION (R IJ S))
 | |
| 
 | |
|             // From each child in children, get the nodes other than commonGrandChildren - these are gc2, gc3, ...
 | |
|             // Each gc2 must be connected by op2 as before, i.e., ABC + ACD = A(BC + CD)
 | |
| 
 | |
|             // All children must be OpCellTreeNodes!
 | |
|             List<OpCellTreeNode> newChildren = new List<OpCellTreeNode>(children.Count);
 | |
|             foreach (OpCellTreeNode child in children)
 | |
|             {
 | |
|                 // Remove all children in child that belong to commonGrandChildren
 | |
|                 // All grandChildren must be leaf nodes at this point
 | |
|                 List<LeafCellTreeNode> newGrandChildren = new List<LeafCellTreeNode>(child.Children.Count);
 | |
|                 foreach (LeafCellTreeNode grandChild in child.Children)
 | |
|                 {
 | |
|                     if (commonGrandChildren.Contains(grandChild) == false)
 | |
|                     {
 | |
|                         newGrandChildren.Add(grandChild);
 | |
|                     }
 | |
|                 }
 | |
|                 // In the above example, child.OpType is IJ
 | |
|                 Debug.Assert(child.OpType == commonChildOpType);
 | |
|                 OpCellTreeNode newChild = new OpCellTreeNode(m_viewgenContext, child.OpType,
 | |
|                                                              Helpers.AsSuperTypeList<LeafCellTreeNode, CellTreeNode>(newGrandChildren));
 | |
|                 newChildren.Add(newChild);
 | |
|             }
 | |
|             // Connect gc2 op1 gc3 op1 gc4 - op1 is UNION in this
 | |
|             // ((X IJ Y) UNION (Y IJ Z) UNION (R IJ S))
 | |
|             // rootNode.Type is UNION
 | |
|             CellTreeNode remainingNodes = new OpCellTreeNode(m_viewgenContext, rootNode.OpType,
 | |
|                                                              Helpers.AsSuperTypeList<OpCellTreeNode, CellTreeNode>(newChildren));
 | |
|             // Take the common grandchildren and connect via commonChildType
 | |
|             // i.e., A IJ B
 | |
|             CellTreeNode commonNodes = new OpCellTreeNode(m_viewgenContext, commonChildOpType,
 | |
|                                                             Helpers.AsSuperTypeList<LeafCellTreeNode, CellTreeNode>(commonGrandChildren));
 | |
| 
 | |
|             // Connect both by commonChildType
 | |
|             CellTreeNode result = new OpCellTreeNode(m_viewgenContext, commonChildOpType,
 | |
|                                                      new CellTreeNode[] { commonNodes, remainingNodes });
 | |
| 
 | |
|             result = result.AssociativeFlatten();
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         // effects: Given a set of nodes, determines if all nodes are the exact same associative opType AND
 | |
|         // there are leaf children that are common across the children "nodes". If there are any,
 | |
|         // returns them. Else return null
 | |
|         private static Set<LeafCellTreeNode> GetCommonGrandChildren(List<CellTreeNode> nodes)
 | |
|         {
 | |
|             Set<LeafCellTreeNode> commonLeaves = null;
 | |
| 
 | |
|             // We could make this general and apply recursively but we don't for now
 | |
| 
 | |
|             // Look for a tree of the form: (common op2 gc2) op1 (common op2 gc3) op1 (common op2 gc4)
 | |
|             // e.g., (A IJ B IJ X IJ Y) UNION (A IJ B IJ Y IJ Z) UNION (A IJ B IJ R IJ S)
 | |
|             // Where op1 and op2 are associative and common, gc2 etc are leaf nodes
 | |
|             CellTreeOpType commonChildOpType = CellTreeOpType.Leaf;
 | |
| 
 | |
|             foreach (CellTreeNode node in nodes)
 | |
|             {
 | |
|                 OpCellTreeNode opNode = node as OpCellTreeNode;
 | |
|                 if (opNode == null)
 | |
|                 {
 | |
|                     return null;
 | |
|                 }
 | |
|                 Debug.Assert(opNode.OpType != CellTreeOpType.Leaf, "Leaf type for op cell node?");
 | |
|                 // Now check for whether the op is associative and the same as the previous one
 | |
|                 if (commonChildOpType == CellTreeOpType.Leaf)
 | |
|                 {
 | |
|                     commonChildOpType = opNode.OpType;
 | |
|                 }
 | |
|                 else if (CellTreeNode.IsAssociativeOp(opNode.OpType) == false || commonChildOpType != opNode.OpType)
 | |
|                 {
 | |
|                     return null;
 | |
|                 }
 | |
| 
 | |
|                 // Make sure all the children are leaf children
 | |
|                 Set<LeafCellTreeNode> nodeChildrenSet = new Set<LeafCellTreeNode>(LeafCellTreeNode.EqualityComparer);
 | |
|                 foreach (CellTreeNode grandChild in opNode.Children)
 | |
|                 {
 | |
|                     LeafCellTreeNode leafGrandChild = grandChild as LeafCellTreeNode;
 | |
|                     if (leafGrandChild == null)
 | |
|                     {
 | |
|                         return null;
 | |
|                     }
 | |
|                     nodeChildrenSet.Add(leafGrandChild);
 | |
|                 }
 | |
| 
 | |
|                 if (commonLeaves == null)
 | |
|                 {
 | |
|                     commonLeaves = nodeChildrenSet;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     commonLeaves.Intersect(nodeChildrenSet);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (commonLeaves.Count == 0)
 | |
|             {
 | |
|                 // No restructuring possible
 | |
|                 return null;
 | |
|             }
 | |
|             return commonLeaves;
 | |
|         }
 | |
|         // effects: Given a list of node, produces a new list in which all
 | |
|         // leaf nodes of the same extent are adjacent to each other. Non-leaf
 | |
|         // nodes are also adjacent to each other. CHANGE_[....]_IMPROVE: Merge with GroupByRightExtent
 | |
|         private static List<CellTreeNode> GroupLeafChildrenByExtent(List<CellTreeNode> nodes)
 | |
|         {
 | |
|             // Keep track of leaf cells for each extent
 | |
|             KeyToListMap<EntitySetBase, CellTreeNode> extentMap =
 | |
|                 new KeyToListMap<EntitySetBase, CellTreeNode>(EqualityComparer<EntitySetBase>.Default);
 | |
| 
 | |
|             List<CellTreeNode> newNodes = new List<CellTreeNode>();
 | |
|             foreach (CellTreeNode node in nodes)
 | |
|             {
 | |
|                 LeafCellTreeNode leafNode = node as LeafCellTreeNode;
 | |
|                 // All non-leaf nodes are added to the result now
 | |
|                 // leaf nodes are added outside the loop
 | |
|                 if (leafNode != null)
 | |
|                 {
 | |
|                     extentMap.Add(leafNode.LeftCellWrapper.RightCellQuery.Extent, leafNode);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     newNodes.Add(node);
 | |
|                 }
 | |
|             }
 | |
|             // Go through the map and add the leaf children
 | |
|             newNodes.AddRange(extentMap.AllValues);
 | |
|             return newNodes;
 | |
|         }
 | |
| 
 | |
|         // effects: A restrictive version of GroupLeafChildrenByExtent --
 | |
|         // only for LASJ and LOJ nodes (works for LOJ only when A LOJ B LOJ C
 | |
|         // s.t., B and C are subsets of A -- in our case that is how LOJs are constructed
 | |
|         private static List<CellTreeNode> GroupNonAssociativeLeafChildren(List<CellTreeNode> nodes)
 | |
|         {
 | |
|             // Keep track of leaf cells for each extent ignoring the 0th child
 | |
|             KeyToListMap<EntitySetBase, CellTreeNode> extentMap =
 | |
|                 new KeyToListMap<EntitySetBase, CellTreeNode>(EqualityComparer<EntitySetBase>.Default);
 | |
| 
 | |
|             List<CellTreeNode> newNodes = new List<CellTreeNode>();
 | |
|             List<CellTreeNode> nonLeafNodes = new List<CellTreeNode>();
 | |
|             // Add the 0th child
 | |
|             newNodes.Add(nodes[0]);
 | |
|             for (int i = 1; i < nodes.Count; i++)
 | |
|             {
 | |
|                 CellTreeNode node = nodes[i];
 | |
|                 LeafCellTreeNode leafNode = node as LeafCellTreeNode;
 | |
|                 // All non-leaf nodes are added to the result now
 | |
|                 // leaf nodes are added outside the loop
 | |
|                 if (leafNode != null)
 | |
|                 {
 | |
|                     extentMap.Add(leafNode.LeftCellWrapper.RightCellQuery.Extent, leafNode);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     nonLeafNodes.Add(node);
 | |
|                 }
 | |
|             }
 | |
|             // Go through the map and add the leaf children
 | |
|             // If a group of nodes exists for the 0th node's extent -- place
 | |
|             // that group first
 | |
|             LeafCellTreeNode firstNode = nodes[0] as LeafCellTreeNode;
 | |
|             if (firstNode != null)
 | |
|             {
 | |
|                 EntitySetBase firstExtent = firstNode.LeftCellWrapper.RightCellQuery.Extent;
 | |
|                 if (extentMap.ContainsKey(firstExtent))
 | |
|                 {
 | |
|                     newNodes.AddRange(extentMap.ListForKey(firstExtent));
 | |
|                     // Remove this set from the map
 | |
|                     extentMap.RemoveKey(firstExtent);
 | |
|                 }
 | |
|             }
 | |
|             newNodes.AddRange(extentMap.AllValues);
 | |
|             newNodes.AddRange(nonLeafNodes);
 | |
|             return newNodes;
 | |
|         }
 | |
| 
 | |
|         // requires: node1 and node2 are two children of the same parent
 | |
|         // connected by opType
 | |
|         // effects: Given two cell tree nodes, node1 and node2, runs the
 | |
|         // TM/SP rule on them to merge them (if they belong to the same
 | |
|         // extent). Returns true if the merge succeeds
 | |
|         private bool TryMergeCellQueries(CellTreeOpType opType, ref CellTreeNode node1,
 | |
|                                          CellTreeNode node2)
 | |
|         {
 | |
| 
 | |
|             LeafCellTreeNode leaf1 = node1 as LeafCellTreeNode;
 | |
|             LeafCellTreeNode leaf2 = node2 as LeafCellTreeNode;
 | |
| 
 | |
|             Debug.Assert(leaf1 != null, "Merge only possible on leaf nodes (1)");
 | |
|             Debug.Assert(leaf2 != null, "Merge only possible on leaf nodes (2)");
 | |
| 
 | |
|             CellQuery mergedLeftCellQuery;
 | |
|             CellQuery mergedRightCellQuery;
 | |
|             if (!TryMergeTwoCellQueries(leaf1.LeftCellWrapper.RightCellQuery, leaf2.LeftCellWrapper.RightCellQuery, opType, m_viewgenContext.MemberMaps.RightDomainMap, out mergedRightCellQuery))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (!TryMergeTwoCellQueries(leaf1.LeftCellWrapper.LeftCellQuery, leaf2.LeftCellWrapper.LeftCellQuery, opType, m_viewgenContext.MemberMaps.LeftDomainMap, out mergedLeftCellQuery))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             // Create a temporary node and add the two children
 | |
|             // so that we can get the merged selectiondomains and attributes
 | |
|             // Note that temp.SelectionDomain below determines the domain
 | |
|             // based on the opType, e.g., for IJ, it intersects the
 | |
|             // multiconstants of all the children
 | |
|             OpCellTreeNode temp = new OpCellTreeNode(m_viewgenContext, opType);
 | |
|             temp.Add(node1);
 | |
|             temp.Add(node2);
 | |
|             // Note: We are losing the original cell number information here and the line number information
 | |
|             // But we will not raise any
 | |
| 
 | |
|             // We do not create CellExpressions with LOJ, FOJ - canBooleansOverlap is true for validation
 | |
|             CellTreeOpType inputOpType = opType;
 | |
|             if (opType == CellTreeOpType.FOJ || opType == CellTreeOpType.LOJ)
 | |
|             {
 | |
|                 inputOpType = CellTreeOpType.IJ;
 | |
|             }
 | |
| 
 | |
|             LeftCellWrapper wrapper = new LeftCellWrapper(m_viewgenContext.ViewTarget, temp.Attributes,
 | |
|                                                           temp.LeftFragmentQuery,
 | |
|                                                           mergedLeftCellQuery,
 | |
|                                                           mergedRightCellQuery,
 | |
|                                                           m_viewgenContext.MemberMaps,
 | |
|                                                           leaf1.LeftCellWrapper.Cells.Concat(leaf2.LeftCellWrapper.Cells));
 | |
|             node1 = new LeafCellTreeNode(m_viewgenContext, wrapper, temp.RightFragmentQuery);
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         // effects: Merges query2 with this according to the TM/SP rules for opType and
 | |
|         // returns the merged result. canBooleansOverlap indicates whether the bools in this and query2 can overlap, i.e.
 | |
|         // the same cells may have contributed to query2 and this earlier in the merge process
 | |
|         internal bool TryMergeTwoCellQueries(CellQuery query1, CellQuery query2, CellTreeOpType opType,
 | |
|                                MemberDomainMap memberDomainMap, out CellQuery mergedQuery)
 | |
|         {
 | |
| 
 | |
|             mergedQuery = null;
 | |
|             // Initialize g1 and g2 according to the TM/SP rules for IJ, LOJ, Union, FOJ cases
 | |
|             BoolExpression g1 = null;
 | |
|             BoolExpression g2 = null;
 | |
|             switch (opType)
 | |
|             {
 | |
|                 case CellTreeOpType.IJ:
 | |
|                     break;
 | |
|                 case CellTreeOpType.LOJ:
 | |
|                 case CellTreeOpType.LASJ:
 | |
|                     g2 = BoolExpression.True;
 | |
|                     break;
 | |
|                 case CellTreeOpType.FOJ:
 | |
|                 case CellTreeOpType.Union:
 | |
|                     g1 = BoolExpression.True;
 | |
|                     g2 = BoolExpression.True;
 | |
|                     break;
 | |
|                 default:
 | |
|                     Debug.Fail("Unsupported operator");
 | |
|                     break;
 | |
|             }
 | |
| 
 | |
|             Dictionary<MemberPath, MemberPath> remap =
 | |
|                 new Dictionary<MemberPath, MemberPath>(MemberPath.EqualityComparer);
 | |
| 
 | |
|             //Continue merging only if both queries are over the same source
 | |
|             MemberPath newRoot;
 | |
|             if (!query1.Extent.Equals(query2.Extent))
 | |
|             { // could not merge
 | |
|                 return false;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 newRoot = query1.SourceExtentMemberPath;
 | |
|             }
 | |
| 
 | |
|             // Conjuncts for ANDing with the previous whereClauses
 | |
|             BoolExpression conjunct1 = BoolExpression.True;
 | |
|             BoolExpression conjunct2 = BoolExpression.True;
 | |
|             BoolExpression whereClause = null;
 | |
| 
 | |
|             switch (opType)
 | |
|             {
 | |
|                 case CellTreeOpType.IJ:
 | |
|                     // Project[D1, D2, A, B, C] Select[cond1 and cond2] (T)
 | |
|                     // We simply merge the two lists of booleans -- no conjuct is added
 | |
|                     // conjunct1 and conjunct2 don't change
 | |
| 
 | |
|                     // query1.WhereCaluse AND query2.WhereCaluse
 | |
|                     Debug.Assert(g1 == null && g2 == null, "IJ does not affect g1 and g2");
 | |
|                     whereClause = BoolExpression.CreateAnd(query1.WhereClause, query2.WhereClause);
 | |
|                     break;
 | |
| 
 | |
|                 case CellTreeOpType.LOJ:
 | |
|                     // conjunct1 does not change since D1 remains as is
 | |
|                     // Project[D1, (expr2 and cond2 and G2) as D2, A, B, C] Select[cond1] (T)
 | |
|                     // D1 does not change. New d2 is the list of booleans expressions
 | |
|                     // for query2 ANDed with g2 AND query2.WhereClause
 | |
|                     Debug.Assert(g1 == null, "LOJ does not affect g1");
 | |
|                     conjunct2 = BoolExpression.CreateAnd(query2.WhereClause, g2);
 | |
|                     // Just query1's whereclause
 | |
|                     whereClause = query1.WhereClause;
 | |
|                     break;
 | |
| 
 | |
|                 case CellTreeOpType.FOJ:
 | |
|                 case CellTreeOpType.Union:
 | |
|                     // Project[(expr1 and cond1 and G1) as D1, (expr2 and cond2 and G2) as D2, A, B, C] Select[cond1] (T)
 | |
|                     // New D1 is a list -- newD1 = D1 AND query1.WhereClause AND g1
 | |
|                     // New D1 is a list -- newD2 = D2 AND query2.WhereClause AND g2
 | |
|                     conjunct1 = BoolExpression.CreateAnd(query1.WhereClause, g1);
 | |
|                     conjunct2 = BoolExpression.CreateAnd(query2.WhereClause, g2);
 | |
| 
 | |
|                     // The new whereClause -- g1 AND query1.WhereCaluse OR g2 AND query2.WhereClause
 | |
|                     whereClause = BoolExpression.CreateOr(BoolExpression.CreateAnd(query1.WhereClause, g1),
 | |
|                                                           BoolExpression.CreateAnd(query2.WhereClause, g2));
 | |
|                     break;
 | |
| 
 | |
|                 case CellTreeOpType.LASJ:
 | |
|                     // conjunct1 does not change since D1 remains as is
 | |
|                     // Project[D1, (expr2 and cond2 and G2) as D2, A, B, C] Select[cond1] (T)
 | |
|                     // D1 does not change. New d2 is the list of booleans expressions
 | |
|                     // for query2 ANDed with g2 AND NOT query2.WhereClause
 | |
|                     Debug.Assert(g1 == null, "LASJ does not affect g1");
 | |
|                     conjunct2 = BoolExpression.CreateAnd(query2.WhereClause, g2);
 | |
|                     whereClause = BoolExpression.CreateAnd(query1.WhereClause, BoolExpression.CreateNot(conjunct2));
 | |
|                     break;
 | |
|                 default:
 | |
|                     Debug.Fail("Unsupported operator");
 | |
|                     break;
 | |
|             }
 | |
| 
 | |
|             // Create the various remapped parts for the cell query --
 | |
|             // boolean expressions, merged slots, whereclause, duplicate
 | |
|             // elimination, join tree
 | |
|             List<BoolExpression> boolExprs =
 | |
|                 MergeBoolExpressions(query1, query2, conjunct1, conjunct2, opType);
 | |
|             //BoolExpression.RemapBools(boolExprs, remap);
 | |
| 
 | |
|             ProjectedSlot[] mergedSlots;
 | |
|             if (false == ProjectedSlot.TryMergeRemapSlots(query1.ProjectedSlots, query2.ProjectedSlots, out mergedSlots))
 | |
|             {
 | |
|                 // merging failed because two different right slots go to same left slot
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             whereClause = whereClause.RemapBool(remap);
 | |
| 
 | |
|             CellQuery.SelectDistinct elimDupl = MergeDupl(query1.SelectDistinctFlag, query2.SelectDistinctFlag);
 | |
| 
 | |
|             whereClause.ExpensiveSimplify();
 | |
|             mergedQuery = new CellQuery(mergedSlots, whereClause,
 | |
|                                                   boolExprs, elimDupl, newRoot);
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         // effects: Given two duplicate eliination choices, returns an OR of them
 | |
|         static private CellQuery.SelectDistinct MergeDupl(CellQuery.SelectDistinct d1, CellQuery.SelectDistinct d2)
 | |
|         {
 | |
|             if (d1 == CellQuery.SelectDistinct.Yes || d2 == CellQuery.SelectDistinct.Yes)
 | |
|             {
 | |
|                 return CellQuery.SelectDistinct.Yes;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return CellQuery.SelectDistinct.No;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // requires: query1 has the same number of boolean expressions as
 | |
|         // query2. There should be no  index i for which query1's bools[i] !=
 | |
|         // null and query2's bools[i] != null
 | |
|         // effects: Given two cellqueries query1 and query2, merges their
 | |
|         // boolean expressions while ANDING query1 bools with conjunct1 and
 | |
|         // query2's bools with conjunct2 and returns the result
 | |
|         static private List<BoolExpression>
 | |
|         MergeBoolExpressions(CellQuery query1, CellQuery query2,
 | |
|                              BoolExpression conjunct1, BoolExpression conjunct2, CellTreeOpType opType)
 | |
|         {
 | |
| 
 | |
|             List<BoolExpression> bools1 = query1.BoolVars;
 | |
|             List<BoolExpression> bools2 = query2.BoolVars;
 | |
| 
 | |
|             // Add conjuncts to both sets if needed
 | |
|             if (false == conjunct1.IsTrue)
 | |
|             {
 | |
|                 bools1 = BoolExpression.AddConjunctionToBools(bools1, conjunct1);
 | |
|             }
 | |
| 
 | |
|             if (false == conjunct2.IsTrue)
 | |
|             {
 | |
|                 bools2 = BoolExpression.AddConjunctionToBools(bools2, conjunct2);
 | |
|             }
 | |
| 
 | |
|             // Perform merge
 | |
|             Debug.Assert(bools1.Count == bools2.Count);
 | |
|             List<BoolExpression> bools = new List<BoolExpression>();
 | |
|             // Both bools1[i] and bools2[i] be null for some of the i's. When
 | |
|             // we merge two (leaf) cells (say), only one boolean each is set
 | |
|             // in it; the rest are all nulls. If the SP/TM rules have been
 | |
|             // applied, more than one boolean may be non-null in a cell query
 | |
|             for (int i = 0; i < bools1.Count; i++)
 | |
|             {
 | |
|                 BoolExpression merged = null;
 | |
|                 if (bools1[i] == null)
 | |
|                 {
 | |
|                     merged = bools2[i];
 | |
|                 }
 | |
|                 else if (bools2[i] == null)
 | |
|                 {
 | |
|                     merged = bools1[i];
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if (opType == CellTreeOpType.IJ)
 | |
|                     {
 | |
|                         merged = BoolExpression.CreateAnd(bools1[i], bools2[i]);
 | |
|                     }
 | |
|                     else if (opType == CellTreeOpType.Union)
 | |
|                     {
 | |
|                         merged = BoolExpression.CreateOr(bools1[i], bools2[i]);
 | |
|                     }
 | |
|                     else if (opType == CellTreeOpType.LASJ)
 | |
|                     {
 | |
|                         merged = BoolExpression.CreateAnd(bools1[i],
 | |
|                                                           BoolExpression.CreateNot(bools2[i]));
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         Debug.Fail("No other operation expected for boolean merge");
 | |
|                     }
 | |
|                 }
 | |
|                 if (merged != null)
 | |
|                 {
 | |
|                     merged.ExpensiveSimplify();
 | |
|                 }
 | |
|                 bools.Add(merged);
 | |
|             }
 | |
|             return bools;
 | |
|         }
 | |
| 
 | |
|         #endregion
 | |
| 
 | |
|         #region String methods
 | |
|         internal override void ToCompactString(StringBuilder builder)
 | |
|         {
 | |
|             m_viewgenContext.MemberMaps.ProjectedSlotMap.ToCompactString(builder);
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|     }
 | |
| }
 |