diff options
| author | Tom Lane <tgl@sss.pgh.pa.us> | 2001-06-05 17:13:52 +0000 | 
|---|---|---|
| committer | Tom Lane <tgl@sss.pgh.pa.us> | 2001-06-05 17:13:52 +0000 | 
| commit | cdd230d62899455cc07ba1caf68387fb834d5bd2 (patch) | |
| tree | c74a0a5342eb883c0265f7caf2993a93e1eff939 /src/backend/optimizer/path/indxpath.c | |
| parent | 7c579fa12df0def35192e1e3cfc9ea7ab90bb0cb (diff) | |
Improve planning of OR indexscan plans: for quals like
	WHERE (a = 1 or a = 2) and b = 42
and an index on (a,b), include the clause b = 42 in the indexquals
generated for each arm of the OR clause.  Essentially this is an index-
driven conversion from CNF to DNF.  Implementation is a bit klugy, but
better than not exploiting the extra quals at all ...
Diffstat (limited to 'src/backend/optimizer/path/indxpath.c')
| -rw-r--r-- | src/backend/optimizer/path/indxpath.c | 109 | 
1 files changed, 71 insertions, 38 deletions
| diff --git a/src/backend/optimizer/path/indxpath.c b/src/backend/optimizer/path/indxpath.c index a5f5bb151da..6a1fd6b50a4 100644 --- a/src/backend/optimizer/path/indxpath.c +++ b/src/backend/optimizer/path/indxpath.c @@ -9,7 +9,7 @@   *   *   * IDENTIFICATION - *	  $Header: /cvsroot/pgsql/src/backend/optimizer/path/indxpath.c,v 1.105 2001/05/20 20:28:18 tgl Exp $ + *	  $Header: /cvsroot/pgsql/src/backend/optimizer/path/indxpath.c,v 1.106 2001/06/05 17:13:51 tgl Exp $   *   *-------------------------------------------------------------------------   */ @@ -397,7 +397,7 @@ match_or_subclause_to_indexkey(RelOptInfo *rel,  										clause, false);  } -/* +/*----------   * Given an OR subclause that has previously been determined to match   * the specified index, extract a list of specific opclauses that can be   * used as indexquals. @@ -406,10 +406,25 @@ match_or_subclause_to_indexkey(RelOptInfo *rel,   * given opclause.	However, if the OR subclause is an AND, we have to   * scan it to find the opclause(s) that match the index.  (There should   * be at least one, if match_or_subclause_to_indexkey succeeded, but there - * could be more.)	Also, we apply expand_indexqual_conditions() to convert - * any special matching opclauses to indexable operators. + * could be more.) + * + * Also, we can look at other restriction clauses of the rel to discover + * additional candidate indexquals: for example, consider + *			... where (a = 11 or a = 12) and b = 42; + * If we are dealing with an index on (a,b) then we can include the clause + * b = 42 in the indexqual list generated for each of the OR subclauses. + * Essentially, we are making an index-specific transformation from CNF to + * DNF.  (NOTE: when we do this, we end up with a slightly inefficient plan + * because create_indexscan_plan is not very bright about figuring out which + * restriction clauses are implied by the generated indexqual condition. + * Currently we'll end up rechecking both the OR clause and the transferred + * restriction clause as qpquals.  FIXME someday.) + * + * Also, we apply expand_indexqual_conditions() to convert any special + * matching opclauses to indexable operators.   *   * The passed-in clause is not changed. + *----------   */  List *  extract_or_indexqual_conditions(RelOptInfo *rel, @@ -417,54 +432,72 @@ extract_or_indexqual_conditions(RelOptInfo *rel,  								Expr *orsubclause)  {  	List	   *quals = NIL; +	int		   *indexkeys = index->indexkeys; +	Oid		   *classes = index->classlist; -	if (and_clause((Node *) orsubclause)) +	/* +	 * Extract relevant indexclauses in indexkey order.  This is essentially +	 * just like group_clauses_by_indexkey() except that the input and +	 * output are lists of bare clauses, not of RestrictInfo nodes. +	 */ +	do  	{ +		int			curIndxKey = indexkeys[0]; +		Oid			curClass = classes[0]; +		List	   *clausegroup = NIL; +		List	   *item; -		/* -		 * Extract relevant sub-subclauses in indexkey order.  This is -		 * just like group_clauses_by_indexkey() except that the input and -		 * output are lists of bare clauses, not of RestrictInfo nodes. -		 */ -		int		   *indexkeys = index->indexkeys; -		Oid		   *classes = index->classlist; +		if (and_clause((Node *) orsubclause)) +		{ +			foreach(item, orsubclause->args) +			{ +				Expr   *subsubclause = (Expr *) lfirst(item); -		do +				if (match_clause_to_indexkey(rel, index, +											 curIndxKey, curClass, +											 subsubclause, false)) +					clausegroup = lappend(clausegroup, subsubclause); +			} +		} +		else if (match_clause_to_indexkey(rel, index, +										  curIndxKey, curClass, +										  orsubclause, false))  		{ -			int			curIndxKey = indexkeys[0]; -			Oid			curClass = classes[0]; -			List	   *clausegroup = NIL; -			List	   *item; +			clausegroup = makeList1(orsubclause); +		} -			foreach(item, orsubclause->args) +		/* +		 * If we found no clauses for this indexkey in the OR subclause +		 * itself, try looking in the rel's top-level restriction list. +		 */ +		if (clausegroup == NIL) +		{ +			foreach(item, rel->baserestrictinfo)  			{ +				RestrictInfo *rinfo = (RestrictInfo *) lfirst(item); +  				if (match_clause_to_indexkey(rel, index,  											 curIndxKey, curClass, -											 lfirst(item), false)) -					clausegroup = lappend(clausegroup, lfirst(item)); +											 rinfo->clause, false)) +					clausegroup = lappend(clausegroup, rinfo->clause);  			} +		} -			/* -			 * If no clauses match this key, we're done; we don't want to -			 * look at keys to its right. -			 */ -			if (clausegroup == NIL) -				break; +		/* +		 * If still no clauses match this key, we're done; we don't want to +		 * look at keys to its right. +		 */ +		if (clausegroup == NIL) +			break; -			quals = nconc(quals, clausegroup); +		quals = nconc(quals, clausegroup); -			indexkeys++; -			classes++; -		} while (!DoneMatchingIndexKeys(indexkeys, index)); +		indexkeys++; +		classes++; +	} while (!DoneMatchingIndexKeys(indexkeys, index)); -		if (quals == NIL) -			elog(ERROR, "extract_or_indexqual_conditions: no matching clause"); -	} -	else -	{ -		/* we assume the caller passed a valid indexable qual */ -		quals = makeList1(orsubclause); -	} +	if (quals == NIL) +		elog(ERROR, "extract_or_indexqual_conditions: no matching clause");  	return expand_indexqual_conditions(quals);  } | 
