<refsynopsisdiv>
 <synopsis>
-CHECKPOINT
+CHECKPOINT [ ( option [, ...] ) ]
+
+<phrase>where <replaceable class="parameter">option</replaceable> can be one of:</phrase>
 </synopsis>
  </refsynopsisdiv>
 
   </para>
  </refsect1>
 
+ <refsect1>
+  <title>Parameters</title>
+
+  <para>
+  </para>
+ </refsect1>
+
  <refsect1>
   <title>Compatibility</title>
 
 
 
                                        $$ = (Node *) n;
                                }
+                       | CHECKPOINT '(' utility_option_list ')'
+                               {
+                                       CheckPointStmt *n = makeNode(CheckPointStmt);
+
+                                       $$ = (Node *) n;
+                                       n->options = $3;
+                               }
                ;
 
 
 
 #include "access/xlog.h"
 #include "access/xlog_internal.h"
 #include "access/xlogrecovery.h"
+#include "catalog/pg_authid.h"
 #include "libpq/pqsignal.h"
 #include "miscadmin.h"
 #include "pgstat.h"
 #include "storage/shmem.h"
 #include "storage/smgr.h"
 #include "storage/spin.h"
+#include "utils/acl.h"
 #include "utils/guc.h"
 #include "utils/memutils.h"
 #include "utils/resowner.h"
        }
 }
 
+/*
+ * ExecCheckpoint
+ *             Primary entry point for manual CHECKPOINT commands
+ *
+ * This is mainly a wrapper for RequestCheckpoint().
+ */
+void
+ExecCheckpoint(ParseState *pstate, CheckPointStmt *stmt)
+{
+       foreach_ptr(DefElem, opt, stmt->options)
+               ereport(ERROR,
+                               (errcode(ERRCODE_SYNTAX_ERROR),
+                                errmsg("unrecognized CHECKPOINT option \"%s\"", opt->defname),
+                                parser_errposition(pstate, opt->location)));
+
+       if (!has_privs_of_role(GetUserId(), ROLE_PG_CHECKPOINT))
+               ereport(ERROR,
+                               (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
+               /* translator: %s is name of an SQL command (e.g., CHECKPOINT) */
+                                errmsg("permission denied to execute %s command",
+                                               "CHECKPOINT"),
+                                errdetail("Only roles with privileges of the \"%s\" role may execute this command.",
+                                                  "pg_checkpoint")));
+
+       RequestCheckpoint(CHECKPOINT_WAIT |
+                                         CHECKPOINT_FAST |
+                                         (RecoveryInProgress() ? 0 : CHECKPOINT_FORCE));
+}
+
 /*
  * RequestCheckpoint
  *             Called in backend processes to request a checkpoint
 
                        break;
 
                case T_CheckPointStmt:
-                       if (!has_privs_of_role(GetUserId(), ROLE_PG_CHECKPOINT))
-                               ereport(ERROR,
-                                               (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
-                               /* translator: %s is name of a SQL command, eg CHECKPOINT */
-                                                errmsg("permission denied to execute %s command",
-                                                               "CHECKPOINT"),
-                                                errdetail("Only roles with privileges of the \"%s\" role may execute this command.",
-                                                                  "pg_checkpoint")));
-
-                       RequestCheckpoint(CHECKPOINT_FAST | CHECKPOINT_WAIT |
-                                                         (RecoveryInProgress() ? 0 : CHECKPOINT_FORCE));
+                       ExecCheckpoint(pstate, (CheckPointStmt *) parsetree);
                        break;
 
                        /*
 
                COMPLETE_WITH_VERSIONED_SCHEMA_QUERY(Query_for_list_of_procedures);
        else if (Matches("CALL", MatchAny))
                COMPLETE_WITH("(");
+/* CHECKPOINT */
+       else if (Matches("CHECKPOINT"))
+               COMPLETE_WITH("(");
 /* CLOSE */
        else if (Matches("CLOSE"))
                COMPLETE_WITH_QUERY_PLUS(Query_for_list_of_cursors,
 
 typedef struct CheckPointStmt
 {
        NodeTag         type;
+       List       *options;            /* list of DefElem nodes */
 } CheckPointStmt;
 
 /* ----------------------
 
 #ifndef _BGWRITER_H
 #define _BGWRITER_H
 
+#include "parser/parse_node.h"
 #include "storage/block.h"
 #include "storage/relfilelocator.h"
 #include "storage/smgr.h"
 pg_noreturn extern void BackgroundWriterMain(const void *startup_data, size_t startup_data_len);
 pg_noreturn extern void CheckpointerMain(const void *startup_data, size_t startup_data_len);
 
+extern void ExecCheckpoint(ParseState *pstate, CheckPointStmt *stmt);
 extern void RequestCheckpoint(int flags);
 extern void CheckpointWriteDelay(int flags, double progress);
 
 
 -- Checkpoint twice: The checkpointer reports stats after reporting completion
 -- of the checkpoint. But after a second checkpoint we'll see at least the
 -- results of the first.
+--
+-- While at it, test checkpoint options.
+CHECKPOINT (WRONG);
+ERROR:  unrecognized CHECKPOINT option "wrong"
+LINE 1: CHECKPOINT (WRONG);
+                    ^
 CHECKPOINT;
 CHECKPOINT;
 SELECT num_requested > :rqst_ckpts_before FROM pg_stat_checkpointer;
 
 -- Checkpoint twice: The checkpointer reports stats after reporting completion
 -- of the checkpoint. But after a second checkpoint we'll see at least the
 -- results of the first.
+--
+-- While at it, test checkpoint options.
+CHECKPOINT (WRONG);
 CHECKPOINT;
 CHECKPOINT;