]> granicus.if.org Git - postgresql/commitdiff
Fix inline_set_returning_function() to allow multiple OUT parameters.
authorTom Lane <tgl@sss.pgh.pa.us>
Thu, 3 Nov 2011 21:53:19 +0000 (17:53 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Thu, 3 Nov 2011 21:53:19 +0000 (17:53 -0400)
inline_set_returning_function failed to distinguish functions returning
generic RECORD (which require a column list in the RTE, as well as run-time
type checking) from those with multiple OUT parameters (which do not).
This prevented inlining from happening.  Per complaint from Jay Levitt.
Back-patch to 8.4 where this capability was introduced.

src/backend/optimizer/util/clauses.c

index 51e0033139c96022507fbed80b728cd085e79eef..fe1a8952fc2694dcf1b757db85114df95949394f 100644 (file)
@@ -26,6 +26,7 @@
 #include "catalog/pg_type.h"
 #include "executor/executor.h"
 #include "executor/functions.h"
+#include "funcapi.h"
 #include "miscadmin.h"
 #include "nodes/makefuncs.h"
 #include "nodes/nodeFuncs.h"
@@ -4476,9 +4477,12 @@ inline_set_returning_function(PlannerInfo *root, RangeTblEntry *rte)
         * If it returns RECORD, we have to check against the column type list
         * provided in the RTE; check_sql_fn_retval can't do that.  (If no match,
         * we just fail to inline, rather than complaining; see notes for
-        * tlist_matches_coltypelist.)
+        * tlist_matches_coltypelist.)  We don't have to do this for functions
+        * with declared OUT parameters, even though their funcresulttype is
+        * RECORDOID, so check get_func_result_type too.
         */
        if (fexpr->funcresulttype == RECORDOID &&
+               get_func_result_type(func_oid, NULL, NULL) == TYPEFUNC_RECORD &&
                !tlist_matches_coltypelist(querytree->targetList, rte->funccoltypes))
                goto fail;