]> granicus.if.org Git - postgresql/commitdiff
Do stack-depth checking in all postmaster children.
authorHeikki Linnakangas <heikki.linnakangas@iki.fi>
Sun, 8 Apr 2012 15:28:12 +0000 (18:28 +0300)
committerHeikki Linnakangas <heikki.linnakangas@iki.fi>
Sun, 8 Apr 2012 16:09:37 +0000 (19:09 +0300)
We used to only initialize the stack base pointer when starting up a regular
backend, not in other processes. In particular, autovacuum workers can run
arbitrary user code, and without stack-depth checking, infinite recursion
in e.g an index expression will bring down the whole cluster.

The comment about PL/Java using set_stack_base() is not yet true. As the
code stands, PL/java still modifies the stack_base_ptr variable directly.
However, it's been discussed in the PL/Java mailing list that it should be
changed to use the function, because PL/Java is currently oblivious to the
register stack used on Itanium. There's another issues with PL/Java, namely
that the stack base pointer it sets is not really the base of the stack, it
could be something close to the bottom of the stack. That's a separate issue
that might need some further changes to this code, but that's a different
story.

Backpatch to all supported releases.

src/backend/postmaster/postmaster.c
src/backend/tcop/postgres.c
src/include/miscadmin.h

index ba51d932d559aa6fd2f5092961db5007ab91d6e3..bbce4446d3bf6abfe094ce587793418eefa6c3b9 100644 (file)
@@ -894,12 +894,18 @@ PostmasterMain(int argc, char *argv[])
         */
        set_max_safe_fds();
 
+       /*
+        * Set reference point for stack-depth checking.
+        */
+       set_stack_base();
+
        /*
         * Load configuration files for client authentication.
         */
        load_hba();
        load_ident();
 
+
        /*
         * Initialize the list of active backends.
         */
@@ -3609,6 +3615,11 @@ SubPostmasterMain(int argc, char *argv[])
        memset(&port, 0, sizeof(Port));
        read_backend_variables(argv[2], &port);
 
+       /*
+        * Set reference point for stack-depth checking
+        */
+       set_stack_base();
+
        /*
         * Set up memory area for GSS information. Mirrors the code in ConnCreate
         * for the non-exec case.
index 10a0dcd9a9d27acfe739913c532427b83254ea18..64b9722cc609bc7ad68a69a29bc8940574fed5a3 100644 (file)
@@ -103,8 +103,10 @@ int                        PostAuthDelay = 0;
 static long max_stack_depth_bytes = 100 * 1024L;
 
 /*
- * Stack base pointer -- initialized by PostgresMain. This is not static
- * so that PL/Java can modify it.
+ * Stack base pointer -- initialized by PostmasterMain and inherited by
+ * subprocesses. This is not static because old versions of PL/Java modify
+ * it directly. Newer versions use set_stack_base(), but we want to stay
+ * binary-compatible for the time being.
  */
 char      *stack_base_ptr = NULL;
 
@@ -2726,6 +2728,53 @@ ia64_get_bsp(void)
 #endif /* IA64 */
 
 
+/*
+ * set_stack_base: set up reference point for stack depth checking
+ *
+ * Returns the old reference point, if any.
+ */
+pg_stack_base_t
+set_stack_base(void)
+{
+       char            stack_base;
+       pg_stack_base_t old;
+
+#if defined(__ia64__) || defined(__ia64)
+       old.stack_base_ptr = stack_base_ptr;
+       old.register_stack_base_ptr = register_stack_base_ptr;
+#else
+       old = stack_base_ptr;
+#endif
+
+       /* Set up reference point for stack depth checking */
+       stack_base_ptr = &stack_base;
+#if defined(__ia64__) || defined(__ia64)
+       register_stack_base_ptr = ia64_get_bsp();
+#endif
+
+       return old;
+}
+
+/*
+ * restore_stack_base: restore reference point for stack depth checking
+ *
+ * This can be used after set_stack_base() to restore the old value. This
+ * is currently only used in PL/Java. When PL/Java calls a backend function
+ * from different thread, the thread's stack is at a different location than
+ * the main thread's stack, so it sets the base pointer before the call, and
+ * restores it afterwards.
+ */
+void
+restore_stack_base(pg_stack_base_t base)
+{
+#if defined(__ia64__) || defined(__ia64)
+       stack_base_ptr = base.stack_base_ptr;
+       register_stack_base_ptr = base.register_stack_base_ptr;
+#else
+       stack_base_ptr = base;
+#endif
+}
+
 /*
  * check_stack_depth: check for excessively deep recursion
  *
@@ -2741,7 +2790,7 @@ check_stack_depth(void)
        long            stack_depth;
 
        /*
-        * Compute distance from PostgresMain's local variables to my own
+        * Compute distance from reference point to to my local variables
         */
        stack_depth = (long) (stack_base_ptr - &stack_top_loc);
 
@@ -2934,7 +2983,6 @@ PostgresMain(int argc, char *argv[], const char *username)
        GucSource       gucsource;
        bool            am_superuser;
        int                     firstchar;
-       char            stack_base;
        StringInfoData input_message;
        sigjmp_buf      local_sigjmp_buf;
        volatile bool send_ready_for_query = true;
@@ -2964,10 +3012,7 @@ PostgresMain(int argc, char *argv[], const char *username)
        SetProcessingMode(InitProcessing);
 
        /* Set up reference point for stack depth checking */
-       stack_base_ptr = &stack_base;
-#if defined(__ia64__) || defined(__ia64)
-       register_stack_base_ptr = ia64_get_bsp();
-#endif
+       set_stack_base();
 
        /* Compute paths, if we didn't inherit them from postmaster */
        if (my_exec_path[0] == '\0')
index a859da28f78f32a6333d9fd71da7c3711960d1a1..34bc1545ae9d886f930fa911e62af46c2f78f9fa 100644 (file)
@@ -219,6 +219,19 @@ extern bool VacuumCostActive;
 
 
 /* in tcop/postgres.c */
+
+#if defined(__ia64__) || defined(__ia64)
+typedef struct
+{
+       char       *stack_base_ptr;
+       char       *register_stack_base_ptr;
+} pg_stack_base_t;
+#else
+typedef char *pg_stack_base_t;
+#endif
+
+extern pg_stack_base_t set_stack_base(void);
+extern void restore_stack_base(pg_stack_base_t base);
 extern void check_stack_depth(void);