Use guc.c's parse_int() instead of pg_atoi() to parse fillfactor in
authorTom Lane <tgl@sss.pgh.pa.us>
Wed, 23 Jul 2008 17:29:53 +0000 (17:29 +0000)
committerTom Lane <tgl@sss.pgh.pa.us>
Wed, 23 Jul 2008 17:29:53 +0000 (17:29 +0000)
default_reloptions().  The previous coding was really a bug because pg_atoi()
will always throw elog on bad input data, whereas default_reloptions is not
supposed to complain about bad input unless its validate parameter is true.
Right now you could only expose the problem by hand-modifying
pg_class.reloptions into an invalid state, so it doesn't seem worth
back-patching; but we should get it right in HEAD because there might be other
situations in future.  Noted while studying GIN fast-update patch.

src/backend/access/common/reloptions.c
src/backend/utils/misc/guc.c
src/include/utils/guc.h

index 532190abe06d5b625734083205ed70d0b59b3310..db39d90d0286f6c1999787de6b2ebda5be0662c7 100644 (file)
@@ -21,6 +21,7 @@
 #include "nodes/makefuncs.h"
 #include "utils/array.h"
 #include "utils/builtins.h"
+#include "utils/guc.h"
 #include "utils/rel.h"
 
 
@@ -287,7 +288,7 @@ default_reloptions(Datum reloptions, bool validate,
 {
        static const char *const default_keywords[1] = {"fillfactor"};
        char       *values[1];
-       int32           fillfactor;
+       int                     fillfactor;
        StdRdOptions *result;
 
        parseRelOptions(reloptions, 1, default_keywords, values, validate);
@@ -300,7 +301,16 @@ default_reloptions(Datum reloptions, bool validate,
        if (values[0] == NULL)
                return NULL;
 
-       fillfactor = pg_atoi(values[0], sizeof(int32), 0);
+       if (!parse_int(values[0], &fillfactor, 0, NULL))
+       {
+               if (validate)
+                       ereport(ERROR,
+                                       (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
+                                        errmsg("fillfactor must be an integer: \"%s\"",
+                                                       values[0])));
+               return NULL;
+       }
+
        if (fillfactor < minFillfactor || fillfactor > 100)
        {
                if (validate)
index 56c42eebb216e5a0f1c7a0446349b60b25cf70ee..1615119b6d4859ce62f3009ca836626729673707 100644 (file)
@@ -4115,7 +4115,7 @@ parse_bool(const char *value, bool *result)
  * If not okay and hintmsg is not NULL, *hintmsg is set to a suitable
  *     HINT message, or NULL if no hint provided.
  */
-static bool
+bool
 parse_int(const char *value, int *result, int flags, const char **hintmsg)
 {
        int64           val;
@@ -4322,7 +4322,7 @@ parse_int(const char *value, int *result, int flags, const char **hintmsg)
  * If the string parses okay, return true, else false.
  * If okay and result is not NULL, return the value in *result.
  */
-static bool
+bool
 parse_real(const char *value, double *result)
 {
        double          val;
index 004df9fdd2fd6a99866ca896f7a68d0459792f7f..5c723b247628cf734e92c9ed6623a230800ca8ed 100644 (file)
@@ -224,6 +224,9 @@ extern void AtEOXact_GUC(bool isCommit, int nestLevel);
 extern void BeginReportingGUCOptions(void);
 extern void ParseLongOption(const char *string, char **name, char **value);
 extern bool parse_bool(const char *value, bool *result);
+extern bool parse_int(const char *value, int *result, int flags,
+                                         const char **hintmsg);
+extern bool parse_real(const char *value, double *result);
 extern bool set_config_option(const char *name, const char *value,
                                  GucContext context, GucSource source,
                                  GucAction action, bool changeVal);