mergeutils.c 3.7 KB
Newer Older
1 2
/*-------------------------------------------------------------------------
 *
3
 * mergeutils.c
4
 *	  Utilities for finding applicable merge clauses and pathkeys
5 6 7 8 9
 *
 * Copyright (c) 1994, Regents of the University of California
 *
 *
 * IDENTIFICATION
10
 *	  $Header: /cvsroot/pgsql/src/backend/optimizer/path/Attic/mergeutils.c,v 1.20 1999/03/01 00:10:32 tgl Exp $
11 12 13 14 15 16 17 18 19 20 21 22 23
 *
 *-------------------------------------------------------------------------
 */
#include "postgres.h"

#include "nodes/pg_list.h"
#include "nodes/relation.h"

#include "optimizer/internal.h"
#include "optimizer/paths.h"
#include "optimizer/clauses.h"
#include "optimizer/ordering.h"

24
/*
25 26
 * group_clauses_by_order
 *	  If a join clause node in 'restrictinfo_list' is mergejoinable, store
27
 *	  it within a mergeinfo node containing other clause nodes with the same
28
 *	  mergejoin ordering.
29
 *
30 31 32 33 34 35 36 37
 * XXX This is completely braindead: there is no reason anymore to segregate
 * mergejoin clauses by join operator, since the executor can handle mergejoin
 * clause sets with different operators in them.  Instead, we ought to be
 * building a MergeInfo for each potentially useful ordering of the input
 * relations.  But right now the optimizer's internal data structures do not
 * support that (MergeInfo can only store one MergeOrder for a set of clauses).
 * Something to fix next time...
 *
38 39
 * 'restrictinfo_list' is the list of restrictinfo nodes
 * 'inner_relid' is the relid of the inner join relation
40
 *
41
 * Returns the new list of mergeinfo nodes.
42
 *
43
 */
44
List *
45
group_clauses_by_order(List *restrictinfo_list,
46
					   int inner_relid)
47
{
48
	List	   *mergeinfo_list = NIL;
49
	List	   *xrestrictinfo;
50

51
	foreach(xrestrictinfo, restrictinfo_list)
52
	{
53 54
		RestrictInfo *restrictinfo = (RestrictInfo *) lfirst(xrestrictinfo);
		MergeOrder *merge_ordering = restrictinfo->mergejoinorder;
55 56 57 58

		if (merge_ordering)
		{
			/*
59
			 * Create a new mergeinfo node and add it to 'mergeinfo_list'
60 61
			 * if one does not yet exist for this merge ordering.
			 */
B
Bruce Momjian 已提交
62
			PathOrder	*pathorder;
63
			MergeInfo	*xmergeinfo;
64
			Expr	   *clause = restrictinfo->clause;
65 66
			Var		   *leftop = get_leftop(clause);
			Var		   *rightop = get_rightop(clause);
67
			JoinKey    *jmkeys;
68

B
Bruce Momjian 已提交
69 70 71 72
			pathorder = makeNode(PathOrder);
			pathorder->ordtype = MERGE_ORDER;
			pathorder->ord.merge = merge_ordering;
			xmergeinfo = match_order_mergeinfo(pathorder, mergeinfo_list);
73 74
			if (inner_relid == leftop->varno)
			{
75 76 77
				jmkeys = makeNode(JoinKey);
				jmkeys->outer = rightop;
				jmkeys->inner = leftop;
78 79 80
			}
			else
			{
81 82 83
				jmkeys = makeNode(JoinKey);
				jmkeys->outer = leftop;
				jmkeys->inner = rightop;
84 85 86 87
			}

			if (xmergeinfo == NULL)
			{
B
Bruce Momjian 已提交
88
				xmergeinfo = makeNode(MergeInfo);
89 90 91 92 93 94

				xmergeinfo->m_ordering = merge_ordering;
				mergeinfo_list = lcons(xmergeinfo,
									   mergeinfo_list);
			}

95 96 97 98
			xmergeinfo->jmethod.clauses = lcons(clause,
												xmergeinfo->jmethod.clauses);
			xmergeinfo->jmethod.jmkeys = lcons(jmkeys,
											   xmergeinfo->jmethod.jmkeys);
99
		}
100
	}
101
	return mergeinfo_list;
102 103 104
}


105
/*
106 107
 * match_order_mergeinfo
 *	  Searches the list 'mergeinfo_list' for a mergeinfo node whose order
108 109
 *	  field equals 'ordering'.
 *
110
 * Returns the node if it exists.
111
 *
112
 */
B
Bruce Momjian 已提交
113
MergeInfo *
114
match_order_mergeinfo(PathOrder *ordering, List *mergeinfo_list)
115
{
116 117
	MergeOrder *xmergeorder;
	List	   *xmergeinfo = NIL;
118

119 120
	foreach(xmergeinfo, mergeinfo_list)
	{
B
Bruce Momjian 已提交
121
		MergeInfo	   *mergeinfo = (MergeInfo *) lfirst(xmergeinfo);
122

123
		xmergeorder = mergeinfo->m_ordering;
124

125
		if ((ordering->ordtype == MERGE_ORDER &&
B
Bruce Momjian 已提交
126
		 equal_merge_ordering(ordering->ord.merge, xmergeorder)) ||
127 128 129
			(ordering->ordtype == SORTOP_ORDER &&
		   equal_path_merge_ordering(ordering->ord.sortop, xmergeorder)))
		{
130

131
			return mergeinfo;
132
		}
133
	}
B
Bruce Momjian 已提交
134
	return (MergeInfo *) NIL;
135
}