*
  *
  * IDENTIFICATION
- *   $PostgreSQL: pgsql/src/backend/optimizer/path/allpaths.c,v 1.151 2006/08/10 02:36:28 tgl Exp $
+ *   $PostgreSQL: pgsql/src/backend/optimizer/path/allpaths.c,v 1.152 2006/08/19 02:48:53 tgl Exp $
  *
  *-------------------------------------------------------------------------
  */
  * 5. We must not push down any quals that refer to subselect outputs that
  * return sets, else we'd introduce functions-returning-sets into the
  * subquery's WHERE/HAVING quals.
+ *
+ * 6. We must not push down any quals that refer to subselect outputs that
+ * contain volatile functions, for fear of introducing strange results due
+ * to multiple evaluation of a volatile function.
  */
 static bool
 qual_is_pushdown_safe(Query *subquery, Index rti, Node *qual,
            safe = false;
            break;
        }
+
+       /* Refuse volatile functions (point 6) */
+       if (contain_volatile_functions((Node *) tle->expr))
+       {
+           safe = false;
+           break;
+       }
    }
 
    list_free(vars);
 
  *
  *
  * IDENTIFICATION
- *   $PostgreSQL: pgsql/src/backend/optimizer/prep/prepjointree.c,v 1.42 2006/08/12 20:05:55 tgl Exp $
+ *   $PostgreSQL: pgsql/src/backend/optimizer/prep/prepjointree.c,v 1.43 2006/08/19 02:48:53 tgl Exp $
  *
  *-------------------------------------------------------------------------
  */
    if (expression_returns_set((Node *) subquery->targetList))
        return false;
 
+   /*
+    * Don't pull up a subquery that has any volatile functions in its
+    * targetlist.  Otherwise we might introduce multiple evaluations of
+    * these functions, if they get copied to multiple places in the upper
+    * query, leading to surprising results.
+    */
+   if (contain_volatile_functions((Node *) subquery->targetList))
+       return false;
+
    /*
     * Hack: don't try to pull up a subquery with an empty jointree.
     * query_planner() will correctly generate a Result plan for a jointree