mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Fix over-optimistic caching in fetch_array_arg_replace_nulls().
When I rewrote this in commit 56a79a869b,
I forgot that it's possible for the input array type to change from one
call to the next (this can happen when applying the function to
pg_statistic columns, for instance).  Fix that.
			
			
This commit is contained in:
		| @@ -28,14 +28,19 @@ static ArrayType * | |||||||
| fetch_array_arg_replace_nulls(FunctionCallInfo fcinfo, int argno) | fetch_array_arg_replace_nulls(FunctionCallInfo fcinfo, int argno) | ||||||
| { | { | ||||||
| 	ArrayType  *v; | 	ArrayType  *v; | ||||||
|  | 	Oid			element_type; | ||||||
| 	ArrayMetaState *my_extra; | 	ArrayMetaState *my_extra; | ||||||
|  |  | ||||||
| 	my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra; | 	/* First collect the array value */ | ||||||
| 	if (my_extra == NULL) | 	if (!PG_ARGISNULL(argno)) | ||||||
| 	{ | 	{ | ||||||
| 		/* First time through, so look up the array type and element type */ | 		v = PG_GETARG_ARRAYTYPE_P(argno); | ||||||
|  | 		element_type = ARR_ELEMTYPE(v); | ||||||
|  | 	} | ||||||
|  | 	else | ||||||
|  | 	{ | ||||||
|  | 		/* We have to look up the array type and element type */ | ||||||
| 		Oid			arr_typeid = get_fn_expr_argtype(fcinfo->flinfo, argno); | 		Oid			arr_typeid = get_fn_expr_argtype(fcinfo->flinfo, argno); | ||||||
| 		Oid			element_type; |  | ||||||
|  |  | ||||||
| 		if (!OidIsValid(arr_typeid)) | 		if (!OidIsValid(arr_typeid)) | ||||||
| 			ereport(ERROR, | 			ereport(ERROR, | ||||||
| @@ -47,26 +52,29 @@ fetch_array_arg_replace_nulls(FunctionCallInfo fcinfo, int argno) | |||||||
| 					(errcode(ERRCODE_DATATYPE_MISMATCH), | 					(errcode(ERRCODE_DATATYPE_MISMATCH), | ||||||
| 					 errmsg("input data type is not an array"))); | 					 errmsg("input data type is not an array"))); | ||||||
|  |  | ||||||
|  | 		v = construct_empty_array(element_type); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/* Now cache required info, which might change from call to call */ | ||||||
|  | 	my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra; | ||||||
|  | 	if (my_extra == NULL) | ||||||
|  | 	{ | ||||||
| 		my_extra = (ArrayMetaState *) | 		my_extra = (ArrayMetaState *) | ||||||
| 			MemoryContextAlloc(fcinfo->flinfo->fn_mcxt, | 			MemoryContextAlloc(fcinfo->flinfo->fn_mcxt, | ||||||
| 							   sizeof(ArrayMetaState)); | 							   sizeof(ArrayMetaState)); | ||||||
| 		my_extra->element_type = element_type; | 		my_extra->element_type = InvalidOid; | ||||||
|  | 		fcinfo->flinfo->fn_extra = my_extra; | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 		/* Cache info about element type */ | 	if (my_extra->element_type != element_type) | ||||||
|  | 	{ | ||||||
| 		get_typlenbyvalalign(element_type, | 		get_typlenbyvalalign(element_type, | ||||||
| 							 &my_extra->typlen, | 							 &my_extra->typlen, | ||||||
| 							 &my_extra->typbyval, | 							 &my_extra->typbyval, | ||||||
| 							 &my_extra->typalign); | 							 &my_extra->typalign); | ||||||
|  | 		my_extra->element_type = element_type; | ||||||
| 		fcinfo->flinfo->fn_extra = my_extra; |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* Now we can collect the array value */ |  | ||||||
| 	if (PG_ARGISNULL(argno)) |  | ||||||
| 		v = construct_empty_array(my_extra->element_type); |  | ||||||
| 	else |  | ||||||
| 		v = PG_GETARG_ARRAYTYPE_P(argno); |  | ||||||
|  |  | ||||||
| 	return v; | 	return v; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user