mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Improve has_nullable_targetlist() to allow strict functions of simple
variables, not just simple variables. This was foreseen in the original coding of this routine, but not implemented until now. Responds to performance gripe from Laurent Perez.
This commit is contained in:
		| @@ -16,7 +16,7 @@ | |||||||
|  * |  * | ||||||
|  * |  * | ||||||
|  * IDENTIFICATION |  * IDENTIFICATION | ||||||
|  *	  $PostgreSQL: pgsql/src/backend/optimizer/prep/prepjointree.c,v 1.15 2004/01/10 00:30:21 tgl Exp $ |  *	  $PostgreSQL: pgsql/src/backend/optimizer/prep/prepjointree.c,v 1.16 2004/01/10 18:13:53 tgl Exp $ | ||||||
|  * |  * | ||||||
|  *------------------------------------------------------------------------- |  *------------------------------------------------------------------------- | ||||||
|  */ |  */ | ||||||
| @@ -439,11 +439,13 @@ is_simple_subquery(Query *subquery) | |||||||
| /* | /* | ||||||
|  * has_nullable_targetlist |  * has_nullable_targetlist | ||||||
|  *	  Check a subquery in the range table to see if all the non-junk |  *	  Check a subquery in the range table to see if all the non-junk | ||||||
|  *	  targetlist items are simple variables (and, hence, will correctly |  *	  targetlist items are simple variables or strict functions of simple | ||||||
|  *	  go to NULL when examined above the point of an outer join). |  *	  variables (and, hence, will correctly go to NULL when examined above | ||||||
|  |  *	  the point of an outer join). | ||||||
|  * |  * | ||||||
|  * A possible future extension is to accept strict functions of simple |  * NOTE: it would be correct (and useful) to ignore output columns that aren't | ||||||
|  * variables, eg, "x + 1". |  * actually referenced by the enclosing query ... but we do not have that | ||||||
|  |  * information available at this point. | ||||||
|  */ |  */ | ||||||
| static bool | static bool | ||||||
| has_nullable_targetlist(Query *subquery) | has_nullable_targetlist(Query *subquery) | ||||||
| @@ -458,11 +460,15 @@ has_nullable_targetlist(Query *subquery) | |||||||
| 		if (tle->resdom->resjunk) | 		if (tle->resdom->resjunk) | ||||||
| 			continue; | 			continue; | ||||||
|  |  | ||||||
| 		/* Okay if tlist item is a simple Var */ | 		/* Must contain a Var of current level */ | ||||||
| 		if (tle->expr && IsA(tle->expr, Var)) | 		if (!contain_vars_of_level((Node *) tle->expr, 0)) | ||||||
| 			continue; | 			return false; | ||||||
|  |  | ||||||
| 		return false; | 		/* Must not contain any non-strict constructs */ | ||||||
|  | 		if (contain_nonstrict_functions((Node *) tle->expr)) | ||||||
|  | 			return false; | ||||||
|  |  | ||||||
|  | 		/* This one's OK, keep scanning */ | ||||||
| 	} | 	} | ||||||
| 	return true; | 	return true; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|  * |  * | ||||||
|  * |  * | ||||||
|  * IDENTIFICATION |  * IDENTIFICATION | ||||||
|  *	  $PostgreSQL: pgsql/src/backend/optimizer/util/clauses.c,v 1.160 2004/01/05 18:04:39 tgl Exp $ |  *	  $PostgreSQL: pgsql/src/backend/optimizer/util/clauses.c,v 1.161 2004/01/10 18:13:53 tgl Exp $ | ||||||
|  * |  * | ||||||
|  * HISTORY |  * HISTORY | ||||||
|  *	  AUTHOR			DATE			MAJOR EVENT |  *	  AUTHOR			DATE			MAJOR EVENT | ||||||
| @@ -697,10 +697,10 @@ contain_volatile_functions_walker(Node *node, void *context) | |||||||
|  * Returns true if any nonstrict construct is found --- ie, anything that |  * Returns true if any nonstrict construct is found --- ie, anything that | ||||||
|  * could produce non-NULL output with a NULL input. |  * could produce non-NULL output with a NULL input. | ||||||
|  * |  * | ||||||
|  * XXX we do not examine sub-selects to see if they contain uses of |  * The idea here is that the caller has verified that the expression contains | ||||||
|  * nonstrict functions. It's not real clear if that is correct or not... |  * one or more Var or Param nodes (as appropriate for the caller's need), and | ||||||
|  * for the current usage it does not matter, since inline_function() |  * now wishes to prove that the expression result will be NULL if any of these | ||||||
|  * rejects cases with sublinks. |  * inputs is NULL.  If we return false, then the proof succeeded. | ||||||
|  */ |  */ | ||||||
| bool | bool | ||||||
| contain_nonstrict_functions(Node *clause) | contain_nonstrict_functions(Node *clause) | ||||||
| @@ -713,6 +713,11 @@ contain_nonstrict_functions_walker(Node *node, void *context) | |||||||
| { | { | ||||||
| 	if (node == NULL) | 	if (node == NULL) | ||||||
| 		return false; | 		return false; | ||||||
|  | 	if (IsA(node, Aggref)) | ||||||
|  | 	{ | ||||||
|  | 		/* an aggregate could return non-null with null input */ | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
| 	if (IsA(node, FuncExpr)) | 	if (IsA(node, FuncExpr)) | ||||||
| 	{ | 	{ | ||||||
| 		FuncExpr   *expr = (FuncExpr *) node; | 		FuncExpr   *expr = (FuncExpr *) node; | ||||||
| @@ -745,16 +750,25 @@ contain_nonstrict_functions_walker(Node *node, void *context) | |||||||
|  |  | ||||||
| 		switch (expr->boolop) | 		switch (expr->boolop) | ||||||
| 		{ | 		{ | ||||||
| 			case OR_EXPR: |  | ||||||
| 			case AND_EXPR: | 			case AND_EXPR: | ||||||
| 				/* OR, AND are inherently non-strict */ | 			case OR_EXPR: | ||||||
|  | 				/* AND, OR are inherently non-strict */ | ||||||
| 				return true; | 				return true; | ||||||
| 			default: | 			default: | ||||||
| 				break; | 				break; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	if (IsA(node, SubLink)) | ||||||
|  | 	{ | ||||||
|  | 		/* In some cases a sublink might be strict, but in general not */ | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 	if (IsA(node, SubPlan)) | ||||||
|  | 		return true; | ||||||
| 	if (IsA(node, CaseExpr)) | 	if (IsA(node, CaseExpr)) | ||||||
| 		return true; | 		return true; | ||||||
|  | 	if (IsA(node, CaseWhen)) | ||||||
|  | 		return true; | ||||||
| 	/* NB: ArrayExpr might someday be nonstrict */ | 	/* NB: ArrayExpr might someday be nonstrict */ | ||||||
| 	if (IsA(node, CoalesceExpr)) | 	if (IsA(node, CoalesceExpr)) | ||||||
| 		return true; | 		return true; | ||||||
| @@ -764,18 +778,6 @@ contain_nonstrict_functions_walker(Node *node, void *context) | |||||||
| 		return true; | 		return true; | ||||||
| 	if (IsA(node, BooleanTest)) | 	if (IsA(node, BooleanTest)) | ||||||
| 		return true; | 		return true; | ||||||
| 	if (IsA(node, SubLink)) |  | ||||||
| 	{ |  | ||||||
| 		SubLink    *sublink = (SubLink *) node; |  | ||||||
| 		List	   *opid; |  | ||||||
|  |  | ||||||
| 		foreach(opid, sublink->operOids) |  | ||||||
| 		{ |  | ||||||
| 			if (!op_strict(lfirsto(opid))) |  | ||||||
| 				return true; |  | ||||||
| 		} |  | ||||||
| 		/* else fall through to check args */ |  | ||||||
| 	} |  | ||||||
| 	return expression_tree_walker(node, contain_nonstrict_functions_walker, | 	return expression_tree_walker(node, contain_nonstrict_functions_walker, | ||||||
| 								  context); | 								  context); | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user