Commit 6b289942 authored by Tom Lane's avatar Tom Lane

Redesign PlanForeignScan API to allow multiple paths for a foreign table.

The original API specification only allowed an FDW to create a single
access path, which doesn't seem like a terribly good idea in hindsight.
Instead, move the responsibility for building the Path node and calling
add_path() into the FDW's PlanForeignScan function.  Now, it can do that
more than once if appropriate.  There is no longer any need for the
transient FdwPlan struct, so get rid of that.

Etsuro Fujita, Shigeru Hanada, Tom Lane
parent 3f47e145
...@@ -25,6 +25,7 @@ ...@@ -25,6 +25,7 @@
#include "miscadmin.h" #include "miscadmin.h"
#include "nodes/makefuncs.h" #include "nodes/makefuncs.h"
#include "optimizer/cost.h" #include "optimizer/cost.h"
#include "optimizer/pathnode.h"
#include "utils/rel.h" #include "utils/rel.h"
#include "utils/syscache.h" #include "utils/syscache.h"
...@@ -93,7 +94,7 @@ PG_FUNCTION_INFO_V1(file_fdw_validator); ...@@ -93,7 +94,7 @@ PG_FUNCTION_INFO_V1(file_fdw_validator);
/* /*
* FDW callback routines * FDW callback routines
*/ */
static FdwPlan *filePlanForeignScan(Oid foreigntableid, static void filePlanForeignScan(Oid foreigntableid,
PlannerInfo *root, PlannerInfo *root,
RelOptInfo *baserel); RelOptInfo *baserel);
static void fileExplainForeignScan(ForeignScanState *node, ExplainState *es); static void fileExplainForeignScan(ForeignScanState *node, ExplainState *es);
...@@ -406,27 +407,44 @@ get_file_fdw_attribute_options(Oid relid) ...@@ -406,27 +407,44 @@ get_file_fdw_attribute_options(Oid relid)
/* /*
* filePlanForeignScan * filePlanForeignScan
* Create a FdwPlan for a scan on the foreign table * Create possible access paths for a scan on the foreign table
*
* Currently we don't support any push-down feature, so there is only one
* possible access path, which simply returns all records in the order in
* the data file.
*/ */
static FdwPlan * static void
filePlanForeignScan(Oid foreigntableid, filePlanForeignScan(Oid foreigntableid,
PlannerInfo *root, PlannerInfo *root,
RelOptInfo *baserel) RelOptInfo *baserel)
{ {
FdwPlan *fdwplan;
char *filename; char *filename;
List *options; List *options;
Cost startup_cost;
Cost total_cost;
/* Fetch options --- we only need filename at this point */ /* Fetch options --- we only need filename at this point */
fileGetOptions(foreigntableid, &filename, &options); fileGetOptions(foreigntableid, &filename, &options);
/* Construct FdwPlan with cost estimates */ /* Estimate costs and update baserel->rows */
fdwplan = makeNode(FdwPlan);
estimate_costs(root, baserel, filename, estimate_costs(root, baserel, filename,
&fdwplan->startup_cost, &fdwplan->total_cost); &startup_cost, &total_cost);
fdwplan->fdw_private = NIL; /* not used */
/* Create a ForeignPath node and add it as only possible path */
add_path(baserel, (Path *)
create_foreignscan_path(root, baserel,
baserel->rows,
startup_cost,
total_cost,
NIL, /* no pathkeys */
NULL, /* no outer rel either */
NIL,
NIL)); /* no fdw_private data */
return fdwplan; /*
* If data file was sorted, and we knew it somehow, we could insert
* appropriate pathkeys into the ForeignPath node to tell the planner that.
*/
} }
/* /*
...@@ -576,6 +594,9 @@ fileReScanForeignScan(ForeignScanState *node) ...@@ -576,6 +594,9 @@ fileReScanForeignScan(ForeignScanState *node)
/* /*
* Estimate costs of scanning a foreign table. * Estimate costs of scanning a foreign table.
*
* In addition to setting *startup_cost and *total_cost, this should
* update baserel->rows.
*/ */
static void static void
estimate_costs(PlannerInfo *root, RelOptInfo *baserel, estimate_costs(PlannerInfo *root, RelOptInfo *baserel,
......
...@@ -88,21 +88,31 @@ ...@@ -88,21 +88,31 @@
<para> <para>
<programlisting> <programlisting>
FdwPlan * void
PlanForeignScan (Oid foreigntableid, PlanForeignScan (Oid foreigntableid,
PlannerInfo *root, PlannerInfo *root,
RelOptInfo *baserel); RelOptInfo *baserel);
</programlisting> </programlisting>
Plan a scan on a foreign table. This is called when a query is planned. Create possible access paths for a scan on a foreign table. This is
called when a query is planned.
<literal>foreigntableid</> is the <structname>pg_class</> OID of the <literal>foreigntableid</> is the <structname>pg_class</> OID of the
foreign table. <literal>root</> is the planner's global information foreign table. <literal>root</> is the planner's global information
about the query, and <literal>baserel</> is the planner's information about the query, and <literal>baserel</> is the planner's information
about this table. about this table.
The function must return a palloc'd struct that contains cost estimates </para>
plus any FDW-private information that is needed to execute the foreign
scan at a later time. (Note that the private information must be <para>
represented in a form that <function>copyObject</> knows how to copy.) The function must generate at least one access path (ForeignPath node)
for a scan on the foreign table and must call <function>add_path</> to
add the path to <literal>baserel-&gt;pathlist</>. It's recommended to
use <function>create_foreignscan_path</> to build the ForeignPath node.
The function may generate multiple access paths, e.g., a path which has
valid <literal>pathkeys</> to represent a pre-sorted result. Each access
path must contain cost estimates, and can contain any FDW-private
information that is needed to execute the foreign scan at a later time.
(Note that the private information must be represented in a form that
<function>copyObject</> knows how to copy.)
</para> </para>
<para> <para>
...@@ -159,9 +169,8 @@ BeginForeignScan (ForeignScanState *node, ...@@ -159,9 +169,8 @@ BeginForeignScan (ForeignScanState *node,
its <structfield>fdw_state</> field is still NULL. Information about its <structfield>fdw_state</> field is still NULL. Information about
the table to scan is accessible through the the table to scan is accessible through the
<structname>ForeignScanState</> node (in particular, from the underlying <structname>ForeignScanState</> node (in particular, from the underlying
<structname>ForeignScan</> plan node, which contains a pointer to the <structname>ForeignScan</> plan node, which contains any FDW-private
<structname>FdwPlan</> structure returned by information provided by <function>PlanForeignScan</>).
<function>PlanForeignScan</>).
</para> </para>
<para> <para>
...@@ -228,9 +237,9 @@ EndForeignScan (ForeignScanState *node); ...@@ -228,9 +237,9 @@ EndForeignScan (ForeignScanState *node);
</para> </para>
<para> <para>
The <structname>FdwRoutine</> and <structname>FdwPlan</> struct types The <structname>FdwRoutine</> struct type is declared in
are declared in <filename>src/include/foreign/fdwapi.h</>, which see <filename>src/include/foreign/fdwapi.h</>, which see for additional
for additional details. details.
</para> </para>
</sect1> </sect1>
......
...@@ -23,7 +23,8 @@ ...@@ -23,7 +23,8 @@
#include "postgres.h" #include "postgres.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "foreign/fdwapi.h" #include "nodes/plannodes.h"
#include "nodes/relation.h"
#include "utils/datum.h" #include "utils/datum.h"
...@@ -591,21 +592,6 @@ _copyForeignScan(const ForeignScan *from) ...@@ -591,21 +592,6 @@ _copyForeignScan(const ForeignScan *from)
* copy remainder of node * copy remainder of node
*/ */
COPY_SCALAR_FIELD(fsSystemCol); COPY_SCALAR_FIELD(fsSystemCol);
COPY_NODE_FIELD(fdwplan);
return newnode;
}
/*
* _copyFdwPlan
*/
static FdwPlan *
_copyFdwPlan(const FdwPlan *from)
{
FdwPlan *newnode = makeNode(FdwPlan);
COPY_SCALAR_FIELD(startup_cost);
COPY_SCALAR_FIELD(total_cost);
COPY_NODE_FIELD(fdw_private); COPY_NODE_FIELD(fdw_private);
return newnode; return newnode;
...@@ -3842,9 +3828,6 @@ copyObject(const void *from) ...@@ -3842,9 +3828,6 @@ copyObject(const void *from)
case T_ForeignScan: case T_ForeignScan:
retval = _copyForeignScan(from); retval = _copyForeignScan(from);
break; break;
case T_FdwPlan:
retval = _copyFdwPlan(from);
break;
case T_Join: case T_Join:
retval = _copyJoin(from); retval = _copyJoin(from);
break; break;
......
...@@ -23,7 +23,9 @@ ...@@ -23,7 +23,9 @@
#include <ctype.h> #include <ctype.h>
#include "foreign/fdwapi.h" #include "lib/stringinfo.h"
#include "nodes/plannodes.h"
#include "nodes/relation.h"
#include "utils/datum.h" #include "utils/datum.h"
...@@ -558,16 +560,6 @@ _outForeignScan(StringInfo str, const ForeignScan *node) ...@@ -558,16 +560,6 @@ _outForeignScan(StringInfo str, const ForeignScan *node)
_outScanInfo(str, (const Scan *) node); _outScanInfo(str, (const Scan *) node);
WRITE_BOOL_FIELD(fsSystemCol); WRITE_BOOL_FIELD(fsSystemCol);
WRITE_NODE_FIELD(fdwplan);
}
static void
_outFdwPlan(StringInfo str, const FdwPlan *node)
{
WRITE_NODE_TYPE("FDWPLAN");
WRITE_FLOAT_FIELD(startup_cost, "%.2f");
WRITE_FLOAT_FIELD(total_cost, "%.2f");
WRITE_NODE_FIELD(fdw_private); WRITE_NODE_FIELD(fdw_private);
} }
...@@ -1572,7 +1564,7 @@ _outForeignPath(StringInfo str, const ForeignPath *node) ...@@ -1572,7 +1564,7 @@ _outForeignPath(StringInfo str, const ForeignPath *node)
_outPathInfo(str, (const Path *) node); _outPathInfo(str, (const Path *) node);
WRITE_NODE_FIELD(fdwplan); WRITE_NODE_FIELD(fdw_private);
} }
static void static void
...@@ -2745,9 +2737,6 @@ _outNode(StringInfo str, const void *obj) ...@@ -2745,9 +2737,6 @@ _outNode(StringInfo str, const void *obj)
case T_ForeignScan: case T_ForeignScan:
_outForeignScan(str, obj); _outForeignScan(str, obj);
break; break;
case T_FdwPlan:
_outFdwPlan(str, obj);
break;
case T_Join: case T_Join:
_outJoin(str, obj); _outJoin(str, obj);
break; break;
......
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include <math.h> #include <math.h>
#include "catalog/pg_class.h" #include "catalog/pg_class.h"
#include "foreign/fdwapi.h"
#include "nodes/nodeFuncs.h" #include "nodes/nodeFuncs.h"
#ifdef OPTIMIZER_DEBUG #ifdef OPTIMIZER_DEBUG
#include "nodes/print.h" #include "nodes/print.h"
...@@ -399,15 +400,18 @@ set_foreign_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte) ...@@ -399,15 +400,18 @@ set_foreign_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte)
/* /*
* set_foreign_pathlist * set_foreign_pathlist
* Build the (single) access path for a foreign table RTE * Build access paths for a foreign table RTE
*/ */
static void static void
set_foreign_pathlist(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte) set_foreign_pathlist(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte)
{ {
/* Generate appropriate path */ FdwRoutine *fdwroutine;
add_path(rel, (Path *) create_foreignscan_path(root, rel));
/* Select cheapest path (pretty easy in this case...) */ /* Call the FDW's PlanForeignScan function to generate path(s) */
fdwroutine = GetFdwRoutineByRelId(rte->relid);
fdwroutine->PlanForeignScan(rte->relid, root, rel);
/* Select cheapest path */
set_cheapest(rel); set_cheapest(rel);
} }
......
...@@ -20,7 +20,6 @@ ...@@ -20,7 +20,6 @@
#include <math.h> #include <math.h>
#include "access/skey.h" #include "access/skey.h"
#include "foreign/fdwapi.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "nodes/makefuncs.h" #include "nodes/makefuncs.h"
#include "nodes/nodeFuncs.h" #include "nodes/nodeFuncs.h"
...@@ -121,7 +120,7 @@ static CteScan *make_ctescan(List *qptlist, List *qpqual, ...@@ -121,7 +120,7 @@ static CteScan *make_ctescan(List *qptlist, List *qpqual,
static WorkTableScan *make_worktablescan(List *qptlist, List *qpqual, static WorkTableScan *make_worktablescan(List *qptlist, List *qpqual,
Index scanrelid, int wtParam); Index scanrelid, int wtParam);
static ForeignScan *make_foreignscan(List *qptlist, List *qpqual, static ForeignScan *make_foreignscan(List *qptlist, List *qpqual,
Index scanrelid, bool fsSystemCol, FdwPlan *fdwplan); Index scanrelid, bool fsSystemCol, List *fdw_private);
static BitmapAnd *make_bitmap_and(List *bitmapplans); static BitmapAnd *make_bitmap_and(List *bitmapplans);
static BitmapOr *make_bitmap_or(List *bitmapplans); static BitmapOr *make_bitmap_or(List *bitmapplans);
static NestLoop *make_nestloop(List *tlist, static NestLoop *make_nestloop(List *tlist,
...@@ -1847,7 +1846,7 @@ create_foreignscan_plan(PlannerInfo *root, ForeignPath *best_path, ...@@ -1847,7 +1846,7 @@ create_foreignscan_plan(PlannerInfo *root, ForeignPath *best_path,
scan_clauses, scan_clauses,
scan_relid, scan_relid,
fsSystemCol, fsSystemCol,
best_path->fdwplan); best_path->fdw_private);
copy_path_costsize(&scan_plan->scan.plan, &best_path->path); copy_path_costsize(&scan_plan->scan.plan, &best_path->path);
...@@ -3189,7 +3188,7 @@ make_foreignscan(List *qptlist, ...@@ -3189,7 +3188,7 @@ make_foreignscan(List *qptlist,
List *qpqual, List *qpqual,
Index scanrelid, Index scanrelid,
bool fsSystemCol, bool fsSystemCol,
FdwPlan *fdwplan) List *fdw_private)
{ {
ForeignScan *node = makeNode(ForeignScan); ForeignScan *node = makeNode(ForeignScan);
Plan *plan = &node->scan.plan; Plan *plan = &node->scan.plan;
...@@ -3201,7 +3200,7 @@ make_foreignscan(List *qptlist, ...@@ -3201,7 +3200,7 @@ make_foreignscan(List *qptlist,
plan->righttree = NULL; plan->righttree = NULL;
node->scan.scanrelid = scanrelid; node->scan.scanrelid = scanrelid;
node->fsSystemCol = fsSystemCol; node->fsSystemCol = fsSystemCol;
node->fdwplan = fdwplan; node->fdw_private = fdw_private;
return node; return node;
} }
......
...@@ -16,7 +16,6 @@ ...@@ -16,7 +16,6 @@
#include <math.h> #include <math.h>
#include "foreign/fdwapi.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "nodes/nodeFuncs.h" #include "nodes/nodeFuncs.h"
#include "optimizer/clauses.h" #include "optimizer/clauses.h"
...@@ -1766,36 +1765,31 @@ create_worktablescan_path(PlannerInfo *root, RelOptInfo *rel) ...@@ -1766,36 +1765,31 @@ create_worktablescan_path(PlannerInfo *root, RelOptInfo *rel)
* create_foreignscan_path * create_foreignscan_path
* Creates a path corresponding to a scan of a foreign table, * Creates a path corresponding to a scan of a foreign table,
* returning the pathnode. * returning the pathnode.
*
* This function is never called from core Postgres; rather, it's expected
* to be called by the PlanForeignScan function of a foreign data wrapper.
* We make the FDW supply all fields of the path, since we do not have any
* way to calculate them in core.
*/ */
ForeignPath * ForeignPath *
create_foreignscan_path(PlannerInfo *root, RelOptInfo *rel) create_foreignscan_path(PlannerInfo *root, RelOptInfo *rel,
double rows, Cost startup_cost, Cost total_cost,
List *pathkeys,
Relids required_outer, List *param_clauses,
List *fdw_private)
{ {
ForeignPath *pathnode = makeNode(ForeignPath); ForeignPath *pathnode = makeNode(ForeignPath);
RangeTblEntry *rte;
FdwRoutine *fdwroutine;
FdwPlan *fdwplan;
pathnode->path.pathtype = T_ForeignScan; pathnode->path.pathtype = T_ForeignScan;
pathnode->path.parent = rel; pathnode->path.parent = rel;
pathnode->path.pathkeys = NIL; /* result is always unordered */ pathnode->path.rows = rows;
pathnode->path.required_outer = NULL; pathnode->path.startup_cost = startup_cost;
pathnode->path.param_clauses = NIL; pathnode->path.total_cost = total_cost;
pathnode->path.pathkeys = pathkeys;
/* Get FDW's callback info */ pathnode->path.required_outer = required_outer;
rte = planner_rt_fetch(rel->relid, root); pathnode->path.param_clauses = param_clauses;
fdwroutine = GetFdwRoutineByRelId(rte->relid);
/* Let the FDW do its planning */
fdwplan = fdwroutine->PlanForeignScan(rte->relid, root, rel);
if (fdwplan == NULL || !IsA(fdwplan, FdwPlan))
elog(ERROR, "foreign-data wrapper PlanForeignScan function for relation %u did not return an FdwPlan struct",
rte->relid);
pathnode->fdwplan = fdwplan;
/* use costs estimated by FDW */ pathnode->fdw_private = fdw_private;
pathnode->path.rows = rel->rows;
pathnode->path.startup_cost = fdwplan->startup_cost;
pathnode->path.total_cost = fdwplan->total_cost;
return pathnode; return pathnode;
} }
......
...@@ -19,40 +19,11 @@ ...@@ -19,40 +19,11 @@
struct ExplainState; struct ExplainState;
/*
* FdwPlan is the information returned to the planner by PlanForeignScan.
*/
typedef struct FdwPlan
{
NodeTag type;
/*
* Cost estimation info. The startup_cost is time before retrieving the
* first row, so it should include costs of connecting to the remote host,
* sending over the query, etc. Note that PlanForeignScan also ought to
* set baserel->rows and baserel->width if it can produce any usable
* estimates of those values.
*/
Cost startup_cost; /* cost expended before fetching any tuples */
Cost total_cost; /* total cost (assuming all tuples fetched) */
/*
* FDW private data, which will be available at execution time.
*
* Note that everything in this list must be copiable by copyObject(). One
* way to store an arbitrary blob of bytes is to represent it as a bytea
* Const. Usually, though, you'll be better off choosing a representation
* that can be dumped usefully by nodeToString().
*/
List *fdw_private;
} FdwPlan;
/* /*
* Callback function signatures --- see fdwhandler.sgml for more info. * Callback function signatures --- see fdwhandler.sgml for more info.
*/ */
typedef FdwPlan *(*PlanForeignScan_function) (Oid foreigntableid, typedef void (*PlanForeignScan_function) (Oid foreigntableid,
PlannerInfo *root, PlannerInfo *root,
RelOptInfo *baserel); RelOptInfo *baserel);
......
...@@ -62,7 +62,6 @@ typedef enum NodeTag ...@@ -62,7 +62,6 @@ typedef enum NodeTag
T_CteScan, T_CteScan,
T_WorkTableScan, T_WorkTableScan,
T_ForeignScan, T_ForeignScan,
T_FdwPlan,
T_Join, T_Join,
T_NestLoop, T_NestLoop,
T_MergeJoin, T_MergeJoin,
......
...@@ -468,8 +468,7 @@ typedef struct ForeignScan ...@@ -468,8 +468,7 @@ typedef struct ForeignScan
{ {
Scan scan; Scan scan;
bool fsSystemCol; /* true if any "system column" is needed */ bool fsSystemCol; /* true if any "system column" is needed */
/* use struct pointer to avoid including fdwapi.h here */ List *fdw_private; /* private data for FDW */
struct FdwPlan *fdwplan;
} ForeignScan; } ForeignScan;
......
...@@ -794,12 +794,18 @@ typedef struct TidPath ...@@ -794,12 +794,18 @@ typedef struct TidPath
/* /*
* ForeignPath represents a scan of a foreign table * ForeignPath represents a scan of a foreign table
*
* fdw_private contains FDW private data about the scan, which will be copied
* to the final ForeignScan plan node so that it is available at execution
* time. Note that everything in this list must be copiable by copyObject().
* One way to store an arbitrary blob of bytes is to represent it as a bytea
* Const. Usually, though, you'll be better off choosing a representation
* that can be dumped usefully by nodeToString().
*/ */
typedef struct ForeignPath typedef struct ForeignPath
{ {
Path path; Path path;
/* use struct pointer to avoid including fdwapi.h here */ List *fdw_private;
struct FdwPlan *fdwplan;
} ForeignPath; } ForeignPath;
/* /*
......
...@@ -68,7 +68,11 @@ extern Path *create_functionscan_path(PlannerInfo *root, RelOptInfo *rel); ...@@ -68,7 +68,11 @@ extern Path *create_functionscan_path(PlannerInfo *root, RelOptInfo *rel);
extern Path *create_valuesscan_path(PlannerInfo *root, RelOptInfo *rel); extern Path *create_valuesscan_path(PlannerInfo *root, RelOptInfo *rel);
extern Path *create_ctescan_path(PlannerInfo *root, RelOptInfo *rel); extern Path *create_ctescan_path(PlannerInfo *root, RelOptInfo *rel);
extern Path *create_worktablescan_path(PlannerInfo *root, RelOptInfo *rel); extern Path *create_worktablescan_path(PlannerInfo *root, RelOptInfo *rel);
extern ForeignPath *create_foreignscan_path(PlannerInfo *root, RelOptInfo *rel); extern ForeignPath *create_foreignscan_path(PlannerInfo *root, RelOptInfo *rel,
double rows, Cost startup_cost, Cost total_cost,
List *pathkeys,
Relids required_outer, List *param_clauses,
List *fdw_private);
extern Relids calc_nestloop_required_outer(Path *outer_path, Path *inner_path); extern Relids calc_nestloop_required_outer(Path *outer_path, Path *inner_path);
extern Relids calc_non_nestloop_required_outer(Path *outer_path, Path *inner_path); extern Relids calc_non_nestloop_required_outer(Path *outer_path, Path *inner_path);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment