aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJason Merrill <jason@redhat.com>2015-06-16 15:29:09 -0400
committerJason Merrill <jason@gcc.gnu.org>2015-06-16 15:29:09 -0400
commit2ed4c0297fb2ece975c69d17e02eac8d8b62325b (patch)
treed77d596beeabd917886f711d575dd0ddf4ed8cf2
parentbd93aa1a976dca96267efb36d6d909ff9813a414 (diff)
downloadgcc-2ed4c0297fb2ece975c69d17e02eac8d8b62325b.zip
gcc-2ed4c0297fb2ece975c69d17e02eac8d8b62325b.tar.gz
gcc-2ed4c0297fb2ece975c69d17e02eac8d8b62325b.tar.bz2
re PR c++/58063 (default arguments evaluated twice per call)
PR c++/58063 * tree.c (bot_manip): Remap SAVE_EXPR. From-SVN: r224533
-rw-r--r--gcc/cp/ChangeLog3
-rw-r--r--gcc/cp/tree.c23
-rw-r--r--gcc/testsuite/g++.dg/overload/defarg10.C28
3 files changed, 54 insertions, 0 deletions
diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog
index 1ab5e8f..5c85c5d 100644
--- a/gcc/cp/ChangeLog
+++ b/gcc/cp/ChangeLog
@@ -1,5 +1,8 @@
2015-06-16 Jason Merrill <jason@redhat.com>
+ PR c++/58063
+ * tree.c (bot_manip): Remap SAVE_EXPR.
+
PR c++/66387
* pt.c (tsubst_copy) [VAR_DECL]: Use process_outer_var_ref.
diff --git a/gcc/cp/tree.c b/gcc/cp/tree.c
index 3553d7c..a52e6f4 100644
--- a/gcc/cp/tree.c
+++ b/gcc/cp/tree.c
@@ -2423,6 +2423,29 @@ bot_manip (tree* tp, int* walk_subtrees, void* data)
*walk_subtrees = 0;
return NULL_TREE;
}
+ if (TREE_CODE (*tp) == SAVE_EXPR)
+ {
+ t = *tp;
+ splay_tree_node n = splay_tree_lookup (target_remap,
+ (splay_tree_key) t);
+ if (n)
+ {
+ *tp = (tree)n->value;
+ *walk_subtrees = 0;
+ }
+ else
+ {
+ copy_tree_r (tp, walk_subtrees, NULL);
+ splay_tree_insert (target_remap,
+ (splay_tree_key)t,
+ (splay_tree_value)*tp);
+ /* Make sure we don't remap an already-remapped SAVE_EXPR. */
+ splay_tree_insert (target_remap,
+ (splay_tree_key)*tp,
+ (splay_tree_value)*tp);
+ }
+ return NULL_TREE;
+ }
/* Make a copy of this node. */
t = copy_tree_r (tp, walk_subtrees, NULL);
diff --git a/gcc/testsuite/g++.dg/overload/defarg10.C b/gcc/testsuite/g++.dg/overload/defarg10.C
new file mode 100644
index 0000000..b8275d6
--- /dev/null
+++ b/gcc/testsuite/g++.dg/overload/defarg10.C
@@ -0,0 +1,28 @@
+// PR c++/58063
+// { dg-do run }
+
+struct basic_ios
+{
+ bool operator!() const { return false; }
+};
+
+struct ostream : virtual basic_ios
+{
+};
+
+int i;
+
+ostream& operator<<(ostream& os, const char* s) {
+ ++i;
+ return os;
+}
+
+ostream cout;
+
+void f(bool x = !(cout << "hi!\n")) { }
+
+int main() {
+ f();
+ if (i != 1)
+ __builtin_abort();
+}