diff --git a/doc/src/sgml/ref/fetch.sgml b/doc/src/sgml/ref/fetch.sgml index 0452cf0144ffb0a76778ab8332835237d022c32a..8f3244eb39ff51fd270f713c6b6a5af8cd109281 100644 --- a/doc/src/sgml/ref/fetch.sgml +++ b/doc/src/sgml/ref/fetch.sgml @@ -1,5 +1,5 @@ <!-- -$Header: /cvsroot/pgsql/doc/src/sgml/ref/fetch.sgml,v 1.26 2003/03/10 03:53:49 tgl Exp $ +$Header: /cvsroot/pgsql/doc/src/sgml/ref/fetch.sgml,v 1.27 2003/03/11 19:40:22 tgl Exp $ PostgreSQL documentation --> @@ -18,17 +18,32 @@ PostgreSQL documentation </refnamediv> <refsynopsisdiv> <refsynopsisdivinfo> - <date>1999-07-20</date> + <date>2003-03-11</date> </refsynopsisdivinfo> <synopsis> -FETCH [ <replaceable class="PARAMETER">direction</replaceable> ] [ <replaceable class="PARAMETER">count</replaceable> ] { IN | FROM } <replaceable class="PARAMETER">cursor</replaceable> -FETCH [ FORWARD | BACKWARD | RELATIVE ] [ <replaceable class="PARAMETER">#</replaceable> | ALL | NEXT | PRIOR ] - { IN | FROM } <replaceable class="PARAMETER">cursor</replaceable> +FETCH [ <replaceable class="PARAMETER">direction</replaceable> { FROM | IN } ] <replaceable class="PARAMETER">cursor</replaceable> + +where <replaceable class="PARAMETER">direction</replaceable> can be empty or one of: + + NEXT + PRIOR + FIRST + LAST + ABSOLUTE <replaceable class="PARAMETER">count</replaceable> + RELATIVE <replaceable class="PARAMETER">count</replaceable> + <replaceable class="PARAMETER">count</replaceable> + ALL + FORWARD + FORWARD <replaceable class="PARAMETER">count</replaceable> + FORWARD ALL + BACKWARD + BACKWARD <replaceable class="PARAMETER">count</replaceable> + BACKWARD ALL </synopsis> <refsect2 id="R2-SQL-FETCH-1"> <refsect2info> - <date>1998-09-01</date> + <date>2003-03-11</date> </refsect2info> <title> Inputs @@ -41,96 +56,170 @@ FETCH [ FORWARD | BACKWARD | RELATIVE ] [ <replaceable class="PARAMETER">#</repl <listitem> <para> <replaceable class="PARAMETER">direction</replaceable> - defines the fetch direction. It can be one of - the following: + defines the fetch direction and number of rows to fetch. + It can be one of the following: <variablelist> <varlistentry> - <term>FORWARD</term> + <term>NEXT</term> <listitem> <para> - fetch next row(s). This is the default + fetch next row. This is the default if <replaceable class="PARAMETER">direction</replaceable> is omitted. </para> </listitem> </varlistentry> + <varlistentry> - <term>BACKWARD</term> + <term>PRIOR</term> <listitem> <para> - fetch previous row(s). + fetch prior row. </para> </listitem> </varlistentry> + <varlistentry> - <term>RELATIVE</term> + <term>FIRST</term> <listitem> <para> - Same as FORWARD; provided for SQL92 compatibility. + fetch first row of query (same as ABSOLUTE 1). </para> </listitem> </varlistentry> - </variablelist> - </para> - </listitem> - </varlistentry> - <varlistentry> - <term><replaceable class="PARAMETER">count</replaceable></term> - <listitem> - <para> - <replaceable class="PARAMETER">count</replaceable> - determines how many rows to fetch. It can be one of the following: + <varlistentry> + <term>LAST</term> + <listitem> + <para> + fetch last row of query (same as ABSOLUTE -1). + </para> + </listitem> + </varlistentry> + + <varlistentry> + <term>ABSOLUTE <replaceable class="PARAMETER">count</replaceable></term> + <listitem> + <para> + fetch the <replaceable class="PARAMETER">count</replaceable>'th + row of query, or the + abs(<replaceable class="PARAMETER">count</replaceable>)'th row + from the end if + <replaceable class="PARAMETER">count</replaceable> < 0. + Position before first row or after last row + if <replaceable class="PARAMETER">count</replaceable> is out of + range; in particular, ABSOLUTE 0 positions before first row. + </para> + </listitem> + </varlistentry> + + <varlistentry> + <term>RELATIVE <replaceable class="PARAMETER">count</replaceable></term> + <listitem> + <para> + fetch the <replaceable class="PARAMETER">count</replaceable>'th + succeeding row, or the + abs(<replaceable class="PARAMETER">count</replaceable>)'th prior + row if <replaceable class="PARAMETER">count</replaceable> < 0. + RELATIVE 0 re-fetches current row, if any. + </para> + </listitem> + </varlistentry> + + <varlistentry> + <term><replaceable class="PARAMETER">count</replaceable></term> + <listitem> + <para> + fetch the next <replaceable class="PARAMETER">count</replaceable> + rows (same as FORWARD <replaceable class="PARAMETER">count</replaceable>). + </para> + </listitem> + </varlistentry> + + <varlistentry> + <term>ALL</term> + <listitem> + <para> + fetch all remaining rows (same as FORWARD ALL). + </para> + </listitem> + </varlistentry> + + <varlistentry> + <term>FORWARD</term> + <listitem> + <para> + fetch next row (same as NEXT). + </para> + </listitem> + </varlistentry> + + <varlistentry> + <term>FORWARD <replaceable class="PARAMETER">count</replaceable></term> + <listitem> + <para> + fetch next <replaceable class="PARAMETER">count</replaceable> + rows. FORWARD 0 re-fetches current row. + </para> + </listitem> + </varlistentry> - <variablelist> <varlistentry> - <term><replaceable class="PARAMETER">#</replaceable></term> + <term>FORWARD ALL</term> <listitem> <para> - A signed integer constant that specifies how many rows to fetch. - Note that a negative integer is equivalent to changing the sense of - FORWARD and BACKWARD. Zero re-fetches the current row, if any. + fetch all remaining rows. </para> </listitem> </varlistentry> <varlistentry> - <term> - ALL - </term> + <term>BACKWARD</term> <listitem> <para> - Retrieve all remaining rows. + fetch prior row (same as PRIOR). </para> </listitem> </varlistentry> <varlistentry> - <term> - NEXT - </term> + <term>BACKWARD <replaceable class="PARAMETER">count</replaceable></term> <listitem> <para> - Equivalent to specifying a count of <command>1</command>. + fetch prior <replaceable class="PARAMETER">count</replaceable> + rows (scanning backwards). BACKWARD 0 re-fetches current row. </para> </listitem> </varlistentry> <varlistentry> - <term> - PRIOR - </term> + <term>BACKWARD ALL</term> <listitem> <para> - Equivalent to specifying a count of <command>-1</command>. + fetch all prior rows (scanning backwards). </para> </listitem> </varlistentry> + </variablelist> </para> </listitem> </varlistentry> + <varlistentry> + <term><replaceable class="PARAMETER">count</replaceable></term> + <listitem> + <para> + <replaceable class="PARAMETER">count</replaceable> + is a possibly-signed integer constant, determining the location + or number of rows to fetch. For FORWARD and BACKWARD cases, + specifying a negative <replaceable + class="PARAMETER">count</replaceable> + is equivalent to changing the sense of FORWARD and BACKWARD. + </para> + </listitem> + </varlistentry> + <varlistentry> <term><replaceable class="PARAMETER">cursor</replaceable></term> <listitem> @@ -145,7 +234,7 @@ FETCH [ FORWARD | BACKWARD | RELATIVE ] [ <replaceable class="PARAMETER">#</repl <refsect2 id="R2-SQL-FETCH-2"> <refsect2info> - <date>1998-04-15</date> + <date>2003-03-11</date> </refsect2info> <title> Outputs @@ -162,25 +251,11 @@ WARNING: PerformPortalFetch: portal "<replaceable class="PARAMETER">cursor</rep </computeroutput></term> <listitem> <para> - If <replaceable class="PARAMETER">cursor</replaceable> - is not previously declared. - The cursor must be declared within a transaction block. + If <replaceable class="PARAMETER">cursor</replaceable> is not known. + The cursor must have been declared within the current transaction block. </para> </listitem> </varlistentry> - - <varlistentry> - <term><computeroutput> -WARNING: FETCH/ABSOLUTE not supported, using RELATIVE - </computeroutput></term> - <listitem> - <para> - <productname>PostgreSQL</productname> does not support absolute - positioning of cursors. - </para> - </listitem> - </varlistentry> - </variablelist> </para> </refsect2> @@ -188,75 +263,79 @@ WARNING: FETCH/ABSOLUTE not supported, using RELATIVE <refsect1 id="R1-SQL-FETCH-1"> <refsect1info> - <date>1998-04-15</date> + <date>2003-03-11</date> </refsect1info> <title> Description </title> <para> - <command>FETCH</command> allows a user to retrieve rows using a cursor. - The number of rows retrieved is specified by - <replaceable class="PARAMETER">#</replaceable>. - If the number of rows remaining in the cursor is less - than <replaceable class="PARAMETER">#</replaceable>, - then only those available are fetched. - Substituting the keyword ALL in place of a number will - cause all remaining rows in the cursor to be retrieved. - Rows may be fetched in both FORWARD and BACKWARD - directions. The default direction is FORWARD. + <command>FETCH</command> retrieves rows using a cursor. </para> <para> - The cursor position can be before the first row of the query result, or on - any particular row of the result, or after the last row of the result. - When created, a cursor is positioned before the first row. After fetching - some rows, the cursor is positioned on the last row retrieved. A new - <command>FETCH</command> always steps one row in the specified direction - (if possible) before beginning to return rows. If the - <command>FETCH</command> requests more rows than available, the cursor is - left positioned after the last row of the query result (or before the first - row, in the case of a backward fetch). This will always be the case after - <command>FETCH ALL</>. + A cursor has an associated <firstterm>position</> that is used by + <command>FETCH</>. The cursor position can be before the first row of the + query result, or on any particular row of the result, or after the last row + of the result. When created, a cursor is positioned before the first row. + After fetching some rows, the cursor is positioned on the row most recently + retrieved. If <command>FETCH</> runs off the end of the available rows + then the cursor is left positioned after the last row, or before the first + row if fetching backward. <command>FETCH ALL</> or <command>FETCH BACKWARD + ALL</> will always leave the cursor positioned after the last row or before + the first row. + </para> + + <para> + The SQL-compatible forms (NEXT, PRIOR, FIRST, LAST, ABSOLUTE, RELATIVE) + fetch a single row after moving the cursor appropriately. If there is + no such row, an empty result is returned, and the cursor is left positioned + before the first row or after the last row as appropriate. + </para> + + <para> + The forms using FORWARD and BACKWARD are not in the SQL standard, but + are <productname>PostgreSQL</productname> extensions. These forms + retrieve the indicated number of rows moving in the forward or backward + direction, leaving the cursor positioned on the last-returned row + (or after/before all rows, if the <replaceable + class="PARAMETER">count</replaceable> exceeds the number of rows + available). </para> <tip> <para> - A zero row count requests fetching the current row without moving the + RELATIVE 0, FORWARD 0, and BACKWARD 0 all request + fetching the current row without moving the cursor --- that is, re-fetching the most recently fetched row. This will succeed unless the cursor is positioned before the first row or after the last row; in which case, no row is returned. </para> </tip> - <tip> - <para> - Negative numbers are allowed to be specified for the - row count. A negative number is equivalent to reversing - the sense of the FORWARD and BACKWARD keywords. For example, - <command>FORWARD -1</command> is the same as <command>BACKWARD 1</command>. - </para> - </tip> - <refsect2 id="R2-SQL-FETCH-3"> <refsect2info> - <date>1998-04-15</date> + <date>2003-03-11</date> </refsect2info> <title> Notes </title> <para> - A cursor to be used in backwards fetching should be declared with the - SCROLL option. In simple cases, <productname>PostgreSQL</productname> - will allow backwards fetch from cursors not declared with SCROLL, but - this behavior is best not relied on. + The cursor should be declared with the SCROLL option if one intends to + use any variants of <command>FETCH</> other than <command>FETCH NEXT</> + or <command>FETCH FORWARD</> with a positive count. For simple queries + <productname>PostgreSQL</productname> will allow backwards fetch from + cursors not declared with SCROLL, but this behavior is best not relied on. </para> <para> - The FORWARD, BACKWARD, and ALL keywords are - <productname>PostgreSQL</productname> extensions. - See below for details on compatibility issues. + ABSOLUTE fetches are not any faster than navigating to the desired row + with a relative move: the underlying implementation must traverse all + the intermediate rows anyway. Negative absolute fetches are even worse: + the query must be read to the end to find the last row, and then + traversed backward from there. However, rewinding to the start of the + query (as with FETCH ABSOLUTE 0) is fast. </para> <para> @@ -316,7 +395,7 @@ FETCH FORWARD 5 IN liahona; </computeroutput> -- Fetch previous row: -FETCH BACKWARD 1 IN liahona; +FETCH PRIOR FROM liahona; <computeroutput> code | title | did | date_prod | kind | len @@ -339,52 +418,39 @@ COMMIT WORK; <refsect2 id="R2-SQL-FETCH-4"> <refsect2info> - <date>1998-09-01</date> + <date>2003-03-11</date> </refsect2info> <title> SQL92 </title> <para> - <note> - <para> - The non-embedded use of cursors is a <productname>PostgreSQL</productname> - extension. The syntax and usage of cursors is being compared - against the embedded form of cursors defined in <acronym>SQL92</acronym>. - </para> - </note> - </para> - - <para> - <acronym>SQL92</acronym> allows absolute positioning of the cursor for - FETCH, and allows placing the results into explicit variables: + <acronym>SQL92</acronym> defines FETCH for use in embedded contexts only. + Therefore, it describes placing the results into explicit variables using + an <literal>INTO</> clause, for example: <synopsis> -FETCH ABSOLUTE <replaceable class="PARAMETER">#</replaceable> +FETCH ABSOLUTE <replaceable class="PARAMETER">n</replaceable> FROM <replaceable class="PARAMETER">cursor</replaceable> INTO :<replaceable class="PARAMETER">variable</replaceable> [, ...] </synopsis> - <variablelist> - <varlistentry> - <term>ABSOLUTE</term> - <listitem> - <para> - The cursor should be positioned to the specified absolute - row number. All row numbers in <productname>PostgreSQL</productname> - are relative numbers so this capability is not supported. - </para> - </listitem> - </varlistentry> - <varlistentry> - <term>:<replaceable class="PARAMETER">variable</replaceable></term> - <listitem> - <para> - Target host variable(s). - </para> - </listitem> - </varlistentry> - </variablelist> + <productname>PostgreSQL</productname>'s use of non-embedded cursors + is non-standard, and so is its practice of returning the result data + as if it were a SELECT result. Other than this point, FETCH is fully + upward-compatible with <acronym>SQL92</acronym>. + </para> + + <para> + The FETCH forms involving FORWARD and BACKWARD (including the forms + FETCH <replaceable class="PARAMETER">count</replaceable> and FETCH ALL, + in which FORWARD is implicit) are <productname>PostgreSQL</productname> + extensions. + </para> + + <para> + <acronym>SQL92</acronym> allows only <literal>FROM</> preceding the + cursor name; the option to use <literal>IN</> is an extension. </para> </refsect2> </refsect1> diff --git a/doc/src/sgml/ref/move.sgml b/doc/src/sgml/ref/move.sgml index 928faabc818bcee4c6696e2e5cf6842a0c3e513a..f01ee9d8a587c4df5a6259347459b2d3609858bf 100644 --- a/doc/src/sgml/ref/move.sgml +++ b/doc/src/sgml/ref/move.sgml @@ -1,5 +1,5 @@ <!-- -$Header: /cvsroot/pgsql/doc/src/sgml/ref/move.sgml,v 1.19 2003/03/10 03:53:49 tgl Exp $ +$Header: /cvsroot/pgsql/doc/src/sgml/ref/move.sgml,v 1.20 2003/03/11 19:40:22 tgl Exp $ PostgreSQL documentation --> @@ -21,7 +21,7 @@ PostgreSQL documentation <date>1999-07-20</date> </refsynopsisdivinfo> <synopsis> -MOVE [ <replaceable class="PARAMETER">direction</replaceable> ] [ <replaceable class="PARAMETER">count</replaceable> ] { IN | FROM } <replaceable class="PARAMETER">cursor</replaceable> +MOVE [ <replaceable class="PARAMETER">direction</replaceable> { FROM | IN } ] <replaceable class="PARAMETER">cursor</replaceable> </synopsis> </refsynopsisdiv> @@ -33,9 +33,7 @@ MOVE [ <replaceable class="PARAMETER">direction</replaceable> ] [ <replaceable c Description </title> <para> - <command>MOVE</command> allows the user to move the cursor position a - specified number of rows, or to the beginning or end of the cursor. - <command>MOVE ALL</command> moves to the end of the cursor. + <command>MOVE</command> repositions a cursor without retrieving any data. <command>MOVE</command> works exactly like the <command>FETCH</command> command, except it only repositions the cursor and does not return rows. </para> @@ -54,8 +52,9 @@ MOVE [ <replaceable class="PARAMETER">direction</replaceable> ] [ <replaceable c </title> <para> - <command>MOVE</command> is a <productname>PostgreSQL</productname> - language extension. + The count returned in <command>MOVE</command>'s status string is the + count of the number of rows that would have been returned by the + equivalent <command>FETCH</command> command. </para> <para> @@ -119,9 +118,6 @@ COMMIT WORK; </title> <para> There is no <acronym>SQL92</acronym> <command>MOVE</command> statement. - Instead, <acronym>SQL92</acronym> allows - one to <command>FETCH</command> rows from an absolute cursor position, - implicitly moving the cursor to the correct position. </para> </refsect2> </refsect1> diff --git a/src/backend/commands/portalcmds.c b/src/backend/commands/portalcmds.c index 0621cdd59036df38dc9acffc72e457a5a3927db7..1ba72437ad72669e35829d0ff6367d45335f0bc1 100644 --- a/src/backend/commands/portalcmds.c +++ b/src/backend/commands/portalcmds.c @@ -8,7 +8,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/commands/portalcmds.c,v 1.9 2003/03/10 03:53:49 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/commands/portalcmds.c,v 1.10 2003/03/11 19:40:22 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -23,6 +23,11 @@ #include "rewrite/rewriteHandler.h" +static long DoRelativeFetch(Portal portal, + bool forward, + long count, + CommandDest dest); +static void DoPortalRewind(Portal portal); static Portal PreparePortal(char *portalName); @@ -102,9 +107,7 @@ PerformCursorOpen(DeclareCursorStmt *stmt, CommandDest dest) * PerformPortalFetch * Execute SQL FETCH or MOVE command. * - * name: name of portal - * forward: forward or backward fetch? - * count: # of tuples to fetch (INT_MAX means "all"; 0 means "refetch") + * stmt: parsetree node for command * dest: where to send results * completionTag: points to a buffer of size COMPLETION_TAG_BUFSIZE * in which to store a command completion status string. @@ -112,9 +115,7 @@ PerformCursorOpen(DeclareCursorStmt *stmt, CommandDest dest) * completionTag may be NULL if caller doesn't want a status string. */ void -PerformPortalFetch(char *name, - bool forward, - long count, +PerformPortalFetch(FetchStmt *stmt, CommandDest dest, char *completionTag) { @@ -123,48 +124,150 @@ PerformPortalFetch(char *name, /* initialize completion status in case of early exit */ if (completionTag) - strcpy(completionTag, (dest == None) ? "MOVE 0" : "FETCH 0"); - - /* sanity checks */ - if (name == NULL) - { - elog(WARNING, "PerformPortalFetch: missing portal name"); - return; - } + strcpy(completionTag, stmt->ismove ? "MOVE 0" : "FETCH 0"); /* get the portal from the portal name */ - portal = GetPortalByName(name); + portal = GetPortalByName(stmt->portalname); if (!PortalIsValid(portal)) { elog(WARNING, "PerformPortalFetch: portal \"%s\" not found", - name); + stmt->portalname); return; } /* Do it */ - nprocessed = DoPortalFetch(portal, forward, count, dest); + nprocessed = DoPortalFetch(portal, + stmt->direction, + stmt->howMany, + stmt->ismove ? None : dest); /* Return command status if wanted */ if (completionTag) snprintf(completionTag, COMPLETION_TAG_BUFSIZE, "%s %ld", - (dest == None) ? "MOVE" : "FETCH", + stmt->ismove ? "MOVE" : "FETCH", nprocessed); } /* * DoPortalFetch - * Guts of PerformPortalFetch --- shared with SPI cursor operations + * Guts of PerformPortalFetch --- shared with SPI cursor operations. + * Caller must already have validated the Portal. * - * Returns number of rows processed. + * Returns number of rows processed (suitable for use in result tag) */ long -DoPortalFetch(Portal portal, bool forward, long count, CommandDest dest) +DoPortalFetch(Portal portal, + FetchDirection fdirection, + long count, + CommandDest dest) { - QueryDesc *queryDesc; - EState *estate; - MemoryContext oldcontext; - ScanDirection direction; - bool temp_desc = false; + bool forward; + + switch (fdirection) + { + case FETCH_FORWARD: + if (count < 0) + { + fdirection = FETCH_BACKWARD; + count = -count; + } + /* fall out of switch to share code with FETCH_BACKWARD */ + break; + case FETCH_BACKWARD: + if (count < 0) + { + fdirection = FETCH_FORWARD; + count = -count; + } + /* fall out of switch to share code with FETCH_FORWARD */ + break; + case FETCH_ABSOLUTE: + if (count > 0) + { + /* + * Definition: Rewind to start, advance count-1 rows, return + * next row (if any). In practice, if the goal is less than + * halfway back to the start, it's better to scan from where + * we are. In any case, we arrange to fetch the target row + * going forwards. + */ + if (portal->posOverflow || portal->portalPos == LONG_MAX || + count-1 <= portal->portalPos / 2) + { + DoPortalRewind(portal); + if (count > 1) + DoRelativeFetch(portal, true, count-1, None); + } + else + { + long pos = portal->portalPos; + + if (portal->atEnd) + pos++; /* need one extra fetch if off end */ + if (count <= pos) + DoRelativeFetch(portal, false, pos-count+1, None); + else if (count > pos+1) + DoRelativeFetch(portal, true, count-pos-1, None); + } + return DoRelativeFetch(portal, true, 1L, dest); + } + else if (count < 0) + { + /* + * Definition: Advance to end, back up abs(count)-1 rows, + * return prior row (if any). We could optimize this if we + * knew in advance where the end was, but typically we won't. + * (Is it worth considering case where count > half of size + * of query? We could rewind once we know the size ...) + */ + DoRelativeFetch(portal, true, FETCH_ALL, None); + if (count < -1) + DoRelativeFetch(portal, false, -count-1, None); + return DoRelativeFetch(portal, false, 1L, dest); + } + else /* count == 0 */ + { + /* Rewind to start, return zero rows */ + DoPortalRewind(portal); + return DoRelativeFetch(portal, true, 0L, dest); + } + break; + case FETCH_RELATIVE: + if (count > 0) + { + /* + * Definition: advance count-1 rows, return next row (if any). + */ + if (count > 1) + DoRelativeFetch(portal, true, count-1, None); + return DoRelativeFetch(portal, true, 1L, dest); + } + else if (count < 0) + { + /* + * Definition: back up abs(count)-1 rows, return prior row + * (if any). + */ + if (count < -1) + DoRelativeFetch(portal, false, -count-1, None); + return DoRelativeFetch(portal, false, 1L, dest); + } + else /* count == 0 */ + { + /* Same as FETCH FORWARD 0, so fall out of switch */ + fdirection = FETCH_FORWARD; + } + break; + default: + elog(ERROR, "DoPortalFetch: bogus direction"); + break; + } + + /* + * Get here with fdirection == FETCH_FORWARD or FETCH_BACKWARD, + * and count >= 0. + */ + forward = (fdirection == FETCH_FORWARD); /* * Zero count means to re-fetch the current row, if any (per SQL92) @@ -174,7 +277,7 @@ DoPortalFetch(Portal portal, bool forward, long count, CommandDest dest) bool on_row; /* Are we sitting on a row? */ - on_row = (portal->atStart == false && portal->atEnd == false); + on_row = (!portal->atStart && !portal->atEnd); if (dest == None) { @@ -187,14 +290,12 @@ DoPortalFetch(Portal portal, bool forward, long count, CommandDest dest) * If we are sitting on a row, back up one so we can re-fetch it. * If we are not sitting on a row, we still have to start up and * shut down the executor so that the destination is initialized - * and shut down correctly; so keep going. Further down in the - * routine, count == 0 means we will retrieve no row. + * and shut down correctly; so keep going. To DoRelativeFetch, + * count == 0 means we will retrieve no row. */ if (on_row) { - DoPortalFetch(portal, - false /* backward */, 1L, - None /* throw away output */); + DoRelativeFetch(portal, false, 1L, None); /* Set up to fetch one row forward */ count = 1; forward = true; @@ -203,9 +304,44 @@ DoPortalFetch(Portal portal, bool forward, long count, CommandDest dest) } /* - * switch into the portal context + * Optimize MOVE BACKWARD ALL into a Rewind. */ - oldcontext = MemoryContextSwitchTo(PortalGetHeapMemory(portal)); + if (!forward && count == FETCH_ALL && dest == None) + { + long result = portal->portalPos; + + if (result > 0 && !portal->atEnd) + result--; + DoPortalRewind(portal); + /* result is bogus if pos had overflowed, but it's best we can do */ + return result; + } + + return DoRelativeFetch(portal, forward, count, dest); +} + +/* + * DoRelativeFetch + * Do fetch for a simple N-rows-forward-or-backward case. + * + * count <= 0 is interpreted as a no-op: the destination gets started up + * and shut down, but nothing else happens. Also, count == FETCH_ALL is + * interpreted as "all rows". + * + * Caller must already have validated the Portal. + * + * Returns number of rows processed (suitable for use in result tag) + */ +static long +DoRelativeFetch(Portal portal, + bool forward, + long count, + CommandDest dest) +{ + QueryDesc *queryDesc; + EState *estate; + ScanDirection direction; + QueryDesc temp_queryDesc; queryDesc = PortalGetQueryDesc(portal); estate = queryDesc->estate; @@ -224,12 +360,9 @@ DoPortalFetch(Portal portal, bool forward, long count, CommandDest dest) if (dest != queryDesc->dest && !(queryDesc->dest == RemoteInternal && dest == Remote)) { - QueryDesc *qdesc = (QueryDesc *) palloc(sizeof(QueryDesc)); - - memcpy(qdesc, queryDesc, sizeof(QueryDesc)); - qdesc->dest = dest; - queryDesc = qdesc; - temp_desc = true; + memcpy(&temp_queryDesc, queryDesc, sizeof(QueryDesc)); + temp_queryDesc.dest = dest; + queryDesc = &temp_queryDesc; } /* @@ -240,65 +373,101 @@ DoPortalFetch(Portal portal, bool forward, long count, CommandDest dest) * robust about being called again if they've already returned NULL * once.) Then call the executor (we must not skip this, because the * destination needs to see a setup and shutdown even if no tuples are - * available). Finally, update the atStart/atEnd state depending on + * available). Finally, update the portal position state depending on * the number of tuples that were retrieved. */ if (forward) { - if (portal->atEnd || count == 0) + if (portal->atEnd || count <= 0) direction = NoMovementScanDirection; else direction = ForwardScanDirection; - /* In the executor, zero count processes all portal rows */ - if (count == INT_MAX) + /* In the executor, zero count processes all rows */ + if (count == FETCH_ALL) count = 0; ExecutorRun(queryDesc, direction, count); if (direction != NoMovementScanDirection) { + long oldPos; + if (estate->es_processed > 0) - portal->atStart = false; /* OK to back up now */ - if (count <= 0 || (long) estate->es_processed < count) + portal->atStart = false; /* OK to go backward now */ + if (count == 0 || + (unsigned long) estate->es_processed < (unsigned long) count) portal->atEnd = true; /* we retrieved 'em all */ + oldPos = portal->portalPos; + portal->portalPos += estate->es_processed; + /* portalPos doesn't advance when we fall off the end */ + if (portal->portalPos < oldPos) + portal->posOverflow = true; } } else { if (!portal->backwardOK) - elog(ERROR, "Cursor cannot scan backwards" + elog(ERROR, "Cursor can only scan forward" "\n\tDeclare it with SCROLL option to enable backward scan"); - if (portal->atStart || count == 0) + if (portal->atStart || count <= 0) direction = NoMovementScanDirection; else direction = BackwardScanDirection; - /* In the executor, zero count processes all portal rows */ - if (count == INT_MAX) + /* In the executor, zero count processes all rows */ + if (count == FETCH_ALL) count = 0; ExecutorRun(queryDesc, direction, count); if (direction != NoMovementScanDirection) { - if (estate->es_processed > 0) + if (estate->es_processed > 0 && portal->atEnd) + { portal->atEnd = false; /* OK to go forward now */ - if (count <= 0 || (long) estate->es_processed < count) + portal->portalPos++; /* adjust for endpoint case */ + } + if (count == 0 || + (unsigned long) estate->es_processed < (unsigned long) count) + { portal->atStart = true; /* we retrieved 'em all */ + portal->portalPos = 0; + portal->posOverflow = false; + } + else + { + long oldPos; + + oldPos = portal->portalPos; + portal->portalPos -= estate->es_processed; + if (portal->portalPos > oldPos || + portal->portalPos <= 0) + portal->posOverflow = true; + } } } - /* - * Clean up and switch back to old context. - */ - if (temp_desc) - pfree(queryDesc); + return estate->es_processed; +} - MemoryContextSwitchTo(oldcontext); +/* + * DoPortalRewind - rewind a Portal to starting point + */ +static void +DoPortalRewind(Portal portal) +{ + QueryDesc *queryDesc; - return estate->es_processed; + queryDesc = PortalGetQueryDesc(portal); + + ExecutorRewind(queryDesc); + + portal->atStart = true; + portal->atEnd = false; + portal->portalPos = 0; + portal->posOverflow = false; } /* @@ -310,15 +479,6 @@ PerformPortalClose(char *name) { Portal portal; - /* - * sanity checks ... why is this case allowed by the grammar, anyway? - */ - if (name == NULL) - { - elog(WARNING, "PerformPortalClose: missing portal name"); - return; - } - /* * get the portal from the portal name */ diff --git a/src/backend/executor/execMain.c b/src/backend/executor/execMain.c index f037e72fd911bba0fe2f458ab09cd35800750abf..638cc61bef4c8bfc98be487f6d26c87998196837 100644 --- a/src/backend/executor/execMain.c +++ b/src/backend/executor/execMain.c @@ -26,7 +26,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/executor/execMain.c,v 1.201 2003/03/10 03:53:49 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/executor/execMain.c,v 1.202 2003/03/11 19:40:22 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -287,6 +287,42 @@ ExecutorEnd(QueryDesc *queryDesc) queryDesc->planstate = NULL; } +/* ---------------------------------------------------------------- + * ExecutorRewind + * + * This routine may be called on an open queryDesc to rewind it + * to the start. + * ---------------------------------------------------------------- + */ +void +ExecutorRewind(QueryDesc *queryDesc) +{ + EState *estate; + MemoryContext oldcontext; + + /* sanity checks */ + Assert(queryDesc != NULL); + + estate = queryDesc->estate; + + Assert(estate != NULL); + + /* It's probably not sensible to rescan updating queries */ + Assert(queryDesc->operation == CMD_SELECT); + + /* + * Switch into per-query memory context + */ + oldcontext = MemoryContextSwitchTo(estate->es_query_cxt); + + /* + * rescan plan + */ + ExecReScan(queryDesc->planstate, NULL); + + MemoryContextSwitchTo(oldcontext); +} + /* * ExecCheckRTPerms diff --git a/src/backend/executor/spi.c b/src/backend/executor/spi.c index e1ccdf08f97425bb69540364c59eef749e15feed..61eed9b4004d11e545a50ea7fb6e7049e284ab53 100644 --- a/src/backend/executor/spi.c +++ b/src/backend/executor/spi.c @@ -8,7 +8,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/executor/spi.c,v 1.87 2003/03/10 03:53:49 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/executor/spi.c,v 1.88 2003/03/11 19:40:22 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -1349,8 +1349,11 @@ _SPI_cursor_operation(Portal portal, bool forward, int count, _SPI_current->tuptable = NULL; /* Run the cursor */ - _SPI_current->processed = DoPortalFetch(portal, forward, (long) count, - dest); + _SPI_current->processed = + DoPortalFetch(portal, + forward ? FETCH_FORWARD : FETCH_BACKWARD, + (long) count, + dest); if (dest == SPI && _SPI_checktuples()) elog(FATAL, "SPI_fetch: # of processed tuples check failed"); diff --git a/src/backend/parser/gram.y b/src/backend/parser/gram.y index 045d3cc2ca24f2e20713526823852a91a89b4b35..7b28bdc9150e45218e3f829c2c5b3ac08a022d76 100644 --- a/src/backend/parser/gram.y +++ b/src/backend/parser/gram.y @@ -11,7 +11,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/parser/gram.y,v 2.405 2003/03/10 03:53:50 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/parser/gram.y,v 2.406 2003/03/11 19:40:23 tgl Exp $ * * HISTORY * AUTHOR DATE MAJOR EVENT @@ -191,7 +191,7 @@ static void doNegateFloat(Value *v); %type <range> qualified_name OptConstrFromTable -%type <str> opt_id all_Op MathOp opt_name SpecialRuleRelation +%type <str> all_Op MathOp opt_name SpecialRuleRelation %type <str> iso_level opt_encoding %type <node> grantee @@ -248,12 +248,10 @@ static void doNegateFloat(Value *v); %type <boolean> copy_from -%type <ival> direction reindex_type drop_type +%type <ival> reindex_type drop_type fetch_count opt_column event comment_type cursor_options -%type <ival> fetch_how_many - -%type <node> select_limit_value select_offset_value +%type <node> fetch_direction select_limit_value select_offset_value %type <list> OptSeqList %type <defelt> OptSeqElem @@ -345,7 +343,7 @@ static void doNegateFloat(Value *v); EACH ELSE ENCODING ENCRYPTED END_P ESCAPE EXCEPT EXCLUSIVE EXECUTE EXISTS EXPLAIN EXTERNAL EXTRACT - FALSE_P FETCH FLOAT_P FOR FORCE FOREIGN FORWARD + FALSE_P FETCH FIRST_P FLOAT_P FOR FORCE FOREIGN FORWARD FREEZE FROM FULL FUNCTION GLOBAL GRANT GROUP_P @@ -361,7 +359,7 @@ static void doNegateFloat(Value *v); KEY - LANCOMPILER LANGUAGE LEADING LEFT LEVEL LIKE LIMIT + LANCOMPILER LANGUAGE LAST_P LEADING LEFT LEVEL LIKE LIMIT LISTEN LOAD LOCAL LOCALTIME LOCALTIMESTAMP LOCATION LOCK_P @@ -1239,16 +1237,15 @@ opt_drop_behavior: ; - /***************************************************************************** * * QUERY : - * close <optname> + * close <portalname> * *****************************************************************************/ ClosePortalStmt: - CLOSE opt_id + CLOSE name { ClosePortalStmt *n = makeNode(ClosePortalStmt); n->portalname = $2; @@ -1256,10 +1253,6 @@ ClosePortalStmt: } ; -opt_id: ColId { $$ = $1; } - | /*EMPTY*/ { $$ = NULL; } - ; - /***************************************************************************** * @@ -2583,151 +2576,159 @@ comment_text: /***************************************************************************** * * QUERY: - * fetch/move [forward | backward] [ # | all ] [ in <portalname> ] - * fetch [ forward | backward | absolute | relative ] - * [ # | all | next | prior ] [ [ in | from ] <portalname> ] + * fetch/move * *****************************************************************************/ -FetchStmt: FETCH direction fetch_how_many from_in name +FetchStmt: FETCH fetch_direction from_in name { - FetchStmt *n = makeNode(FetchStmt); - if ($3 < 0) - { - $3 = -$3; - $2 = (($2 == FETCH_FORWARD) ? FETCH_BACKWARD : FETCH_FORWARD); - } - n->direction = $2; - n->howMany = $3; - n->portalname = $5; + FetchStmt *n = (FetchStmt *) $2; + n->portalname = $4; n->ismove = FALSE; $$ = (Node *)n; } - | FETCH fetch_how_many from_in name + | FETCH name { FetchStmt *n = makeNode(FetchStmt); - if ($2 < 0) - { - n->howMany = -$2; - n->direction = FETCH_BACKWARD; - } - else - { - n->direction = FETCH_FORWARD; - n->howMany = $2; - } - n->portalname = $4; + n->direction = FETCH_FORWARD; + n->howMany = 1; + n->portalname = $2; n->ismove = FALSE; $$ = (Node *)n; } - | FETCH direction from_in name + | MOVE fetch_direction from_in name { - FetchStmt *n = makeNode(FetchStmt); - n->direction = $2; - n->howMany = 1; + FetchStmt *n = (FetchStmt *) $2; n->portalname = $4; - n->ismove = FALSE; + n->ismove = TRUE; $$ = (Node *)n; } - | FETCH from_in name + | MOVE name { FetchStmt *n = makeNode(FetchStmt); n->direction = FETCH_FORWARD; n->howMany = 1; - n->portalname = $3; - n->ismove = FALSE; + n->portalname = $2; + n->ismove = TRUE; $$ = (Node *)n; } - | FETCH name + ; + +fetch_direction: + /*EMPTY*/ { FetchStmt *n = makeNode(FetchStmt); n->direction = FETCH_FORWARD; n->howMany = 1; - n->portalname = $2; - n->ismove = FALSE; $$ = (Node *)n; } - | MOVE direction fetch_how_many from_in name + | NEXT { FetchStmt *n = makeNode(FetchStmt); - if ($3 < 0) - { - $3 = -$3; - $2 = (($2 == FETCH_FORWARD) ? FETCH_BACKWARD : FETCH_FORWARD); - } - n->direction = $2; - n->howMany = $3; - n->portalname = $5; - n->ismove = TRUE; + n->direction = FETCH_FORWARD; + n->howMany = 1; $$ = (Node *)n; } - | MOVE fetch_how_many from_in name + | PRIOR { FetchStmt *n = makeNode(FetchStmt); - if ($2 < 0) - { - n->howMany = -$2; - n->direction = FETCH_BACKWARD; - } - else - { - n->direction = FETCH_FORWARD; - n->howMany = $2; - } - n->portalname = $4; - n->ismove = TRUE; + n->direction = FETCH_BACKWARD; + n->howMany = 1; $$ = (Node *)n; } - | MOVE direction from_in name + | FIRST_P { FetchStmt *n = makeNode(FetchStmt); - n->direction = $2; + n->direction = FETCH_ABSOLUTE; n->howMany = 1; - n->portalname = $4; - n->ismove = TRUE; $$ = (Node *)n; } - | MOVE from_in name + | LAST_P + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_ABSOLUTE; + n->howMany = -1; + $$ = (Node *)n; + } + | ABSOLUTE fetch_count + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_ABSOLUTE; + n->howMany = $2; + $$ = (Node *)n; + } + | RELATIVE fetch_count + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_RELATIVE; + n->howMany = $2; + $$ = (Node *)n; + } + | fetch_count + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_FORWARD; + n->howMany = $1; + $$ = (Node *)n; + } + | ALL + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_FORWARD; + n->howMany = FETCH_ALL; + $$ = (Node *)n; + } + | FORWARD { FetchStmt *n = makeNode(FetchStmt); n->direction = FETCH_FORWARD; n->howMany = 1; - n->portalname = $3; - n->ismove = TRUE; $$ = (Node *)n; } - | MOVE name + | FORWARD fetch_count { FetchStmt *n = makeNode(FetchStmt); n->direction = FETCH_FORWARD; + n->howMany = $2; + $$ = (Node *)n; + } + | FORWARD ALL + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_FORWARD; + n->howMany = FETCH_ALL; + $$ = (Node *)n; + } + | BACKWARD + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_BACKWARD; n->howMany = 1; - n->portalname = $2; - n->ismove = TRUE; $$ = (Node *)n; } - ; - -direction: FORWARD { $$ = FETCH_FORWARD; } - | BACKWARD { $$ = FETCH_BACKWARD; } - | RELATIVE { $$ = FETCH_FORWARD; } - | ABSOLUTE + | BACKWARD fetch_count + { + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_BACKWARD; + n->howMany = $2; + $$ = (Node *)n; + } + | BACKWARD ALL { - elog(NOTICE, - "FETCH / ABSOLUTE not supported, using RELATIVE"); - $$ = FETCH_FORWARD; + FetchStmt *n = makeNode(FetchStmt); + n->direction = FETCH_BACKWARD; + n->howMany = FETCH_ALL; + $$ = (Node *)n; } ; -fetch_how_many: +fetch_count: Iconst { $$ = $1; } | '-' Iconst { $$ = - $2; } - | ALL { $$ = INT_MAX; } - | NEXT { $$ = 1; } - | PRIOR { $$ = -1; } ; -from_in: IN_P {} - | FROM {} +from_in: FROM {} + | IN_P {} ; @@ -7093,6 +7094,7 @@ unreserved_keyword: | EXPLAIN | EXTERNAL | FETCH + | FIRST_P | FORCE | FORWARD | FUNCTION @@ -7115,6 +7117,7 @@ unreserved_keyword: | KEY | LANCOMPILER | LANGUAGE + | LAST_P | LEVEL | LISTEN | LOAD @@ -7170,9 +7173,9 @@ unreserved_keyword: | SCROLL | SECOND_P | SECURITY - | SESSION | SEQUENCE | SERIALIZABLE + | SESSION | SET | SHARE | SHOW @@ -7211,8 +7214,8 @@ unreserved_keyword: | VOLATILE | WITH | WITHOUT - | WRITE | WORK + | WRITE | YEAR_P | ZONE ; diff --git a/src/backend/parser/keywords.c b/src/backend/parser/keywords.c index 727cff3eaf4bfeb37ca7ffa24823db3105048d70..49432cb957a83f0a44dc0843d6356037bc253ca1 100644 --- a/src/backend/parser/keywords.c +++ b/src/backend/parser/keywords.c @@ -8,7 +8,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/parser/keywords.c,v 1.134 2003/02/10 04:44:46 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/parser/keywords.c,v 1.135 2003/03/11 19:40:23 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -128,6 +128,7 @@ static const ScanKeyword ScanKeywords[] = { {"extract", EXTRACT}, {"false", FALSE_P}, {"fetch", FETCH}, + {"first", FIRST_P}, {"float", FLOAT_P}, {"for", FOR}, {"force", FORCE}, @@ -171,6 +172,7 @@ static const ScanKeyword ScanKeywords[] = { {"key", KEY}, {"lancompiler", LANCOMPILER}, {"language", LANGUAGE}, + {"last", LAST_P}, {"leading", LEADING}, {"left", LEFT}, {"level", LEVEL}, diff --git a/src/backend/tcop/utility.c b/src/backend/tcop/utility.c index 0fae711a2c320930aaae5fef4583e78929e48a06..baf74aa3cc5de21757ac85ec46b862c3df8a189c 100644 --- a/src/backend/tcop/utility.c +++ b/src/backend/tcop/utility.c @@ -10,7 +10,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/tcop/utility.c,v 1.194 2003/03/10 03:53:51 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/tcop/utility.c,v 1.195 2003/03/11 19:40:23 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -321,15 +321,8 @@ ProcessUtility(Node *parsetree, break; case T_FetchStmt: - { - FetchStmt *stmt = (FetchStmt *) parsetree; - - PerformPortalFetch(stmt->portalname, - stmt->direction == FETCH_FORWARD, - stmt->howMany, - (stmt->ismove) ? None : dest, - completionTag); - } + PerformPortalFetch((FetchStmt *) parsetree, dest, + completionTag); break; /* diff --git a/src/backend/utils/mmgr/portalmem.c b/src/backend/utils/mmgr/portalmem.c index 654247dd8c5a848f7157079065714ade58c1aa56..66ee72718cfa640baf2872608181c4dfd9acc574 100644 --- a/src/backend/utils/mmgr/portalmem.c +++ b/src/backend/utils/mmgr/portalmem.c @@ -8,7 +8,7 @@ * * * IDENTIFICATION - * $Header: /cvsroot/pgsql/src/backend/utils/mmgr/portalmem.c,v 1.52 2003/03/10 03:53:51 tgl Exp $ + * $Header: /cvsroot/pgsql/src/backend/utils/mmgr/portalmem.c,v 1.53 2003/03/11 19:40:23 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -167,10 +167,12 @@ PortalSetQuery(Portal portal, AssertArg(PortalIsValid(portal)); portal->queryDesc = queryDesc; - portal->backwardOK = ExecSupportsBackwardScan(queryDesc->plantree); - portal->atStart = true; /* Allow fetch forward only, to start */ - portal->atEnd = false; portal->cleanup = cleanup; + portal->backwardOK = ExecSupportsBackwardScan(queryDesc->plantree); + portal->atStart = true; + portal->atEnd = false; /* allow fetches */ + portal->portalPos = 0; + portal->posOverflow = false; } /* @@ -211,10 +213,12 @@ CreatePortal(const char *name) /* initialize portal query */ portal->queryDesc = NULL; - portal->backwardOK = false; - portal->atStart = true; /* disallow fetches until query is set */ - portal->atEnd = true; portal->cleanup = NULL; + portal->backwardOK = false; + portal->atStart = true; + portal->atEnd = true; /* disallow fetches until query is set */ + portal->portalPos = 0; + portal->posOverflow = false; /* put portal in table */ PortalHashTableInsert(portal); diff --git a/src/include/commands/portalcmds.h b/src/include/commands/portalcmds.h index 3f2a4221add76007db73e7b07fc2ea6276649338..74855ddf60545c2d8f29bd03f99c9725ac3bcfff 100644 --- a/src/include/commands/portalcmds.h +++ b/src/include/commands/portalcmds.h @@ -7,7 +7,7 @@ * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * - * $Id: portalcmds.h,v 1.5 2003/03/10 03:53:51 tgl Exp $ + * $Id: portalcmds.h,v 1.6 2003/03/11 19:40:23 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -19,10 +19,12 @@ extern void PerformCursorOpen(DeclareCursorStmt *stmt, CommandDest dest); -extern void PerformPortalFetch(char *name, bool forward, long count, - CommandDest dest, char *completionTag); +extern void PerformPortalFetch(FetchStmt *stmt, CommandDest dest, + char *completionTag); -extern long DoPortalFetch(Portal portal, bool forward, long count, +extern long DoPortalFetch(Portal portal, + FetchDirection fdirection, + long count, CommandDest dest); extern void PerformPortalClose(char *name); diff --git a/src/include/executor/executor.h b/src/include/executor/executor.h index 785d21718b2b5328e8ddaaf39a6efecef73aab60..40a696e2976459a894de4164876caa25a5cf3d80 100644 --- a/src/include/executor/executor.h +++ b/src/include/executor/executor.h @@ -7,7 +7,7 @@ * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * - * $Id: executor.h,v 1.90 2003/03/10 03:53:51 tgl Exp $ + * $Id: executor.h,v 1.91 2003/03/11 19:40:23 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -86,6 +86,7 @@ extern void ExecutorStart(QueryDesc *queryDesc); extern TupleTableSlot *ExecutorRun(QueryDesc *queryDesc, ScanDirection direction, long count); extern void ExecutorEnd(QueryDesc *queryDesc); +extern void ExecutorRewind(QueryDesc *queryDesc); extern void ExecCheckRTPerms(List *rangeTable, CmdType operation); extern void ExecEndPlan(PlanState *planstate, EState *estate); extern void ExecConstraints(const char *caller, ResultRelInfo *resultRelInfo, diff --git a/src/include/nodes/parsenodes.h b/src/include/nodes/parsenodes.h index c84348ded9e85383ed7ce95690866550420dc8b4..216ed04c7624a1f74390ea68e52441e3f35da5eb 100644 --- a/src/include/nodes/parsenodes.h +++ b/src/include/nodes/parsenodes.h @@ -7,7 +7,7 @@ * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * - * $Id: parsenodes.h,v 1.232 2003/03/10 03:53:51 tgl Exp $ + * $Id: parsenodes.h,v 1.233 2003/03/11 19:40:23 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -1228,16 +1228,21 @@ typedef struct ClosePortalStmt */ typedef enum FetchDirection { + /* for these, howMany is how many rows to fetch; FETCH_ALL means ALL */ FETCH_FORWARD, - FETCH_BACKWARD - /* ABSOLUTE someday? */ + FETCH_BACKWARD, + /* for these, howMany indicates a position; only one row is fetched */ + FETCH_ABSOLUTE, + FETCH_RELATIVE } FetchDirection; +#define FETCH_ALL LONG_MAX + typedef struct FetchStmt { NodeTag type; FetchDirection direction; /* see above */ - long howMany; /* number of rows */ + long howMany; /* number of rows, or position argument */ char *portalname; /* name of portal (cursor) */ bool ismove; /* TRUE if MOVE */ } FetchStmt; diff --git a/src/include/utils/portal.h b/src/include/utils/portal.h index 21469dd52df442631cb6632f10e96a25acaf09af..c9ca8547ce2d773dfcc69eec2e571d4c56967811 100644 --- a/src/include/utils/portal.h +++ b/src/include/utils/portal.h @@ -9,7 +9,7 @@ * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * - * $Id: portal.h,v 1.38 2003/03/10 03:53:52 tgl Exp $ + * $Id: portal.h,v 1.39 2003/03/11 19:40:24 tgl Exp $ * *------------------------------------------------------------------------- */ @@ -27,10 +27,21 @@ typedef struct PortalData char *name; /* Portal's name */ MemoryContext heap; /* subsidiary memory */ QueryDesc *queryDesc; /* Info about query associated with portal */ - bool backwardOK; /* is fetch backwards allowed at all? */ - bool atStart; /* T => fetch backwards is not allowed now */ - bool atEnd; /* T => fetch forwards is not allowed now */ void (*cleanup) (Portal); /* Cleanup routine (optional) */ + bool backwardOK; /* is fetch backwards allowed? */ + /* + * atStart, atEnd and portalPos indicate the current cursor position. + * portalPos is zero before the first row, N after fetching N'th row of + * query. After we run off the end, portalPos = # of rows in query, and + * atEnd is true. If portalPos overflows, set posOverflow (this causes + * us to stop relying on its value for navigation). Note that atStart + * implies portalPos == 0, but not the reverse (portalPos could have + * overflowed). + */ + bool atStart; + bool atEnd; + bool posOverflow; + long portalPos; } PortalData; /*