We were calling fold_non_dependent_expr twice on the template argument,
which leads to chaos. In general we try to fold close to the use, so
let's stop folding in the parser.
Tested x86_64-pc-linux-gnu, applying to trunk.
commit 4240d0bfcd28f4a822ae0fff3f045e2eda46f55c
Author: Jason Merrill <ja...@redhat.com>
Date: Tue Feb 12 23:40:45 2013 -0500
PR c++/55931
* parser.c (cp_parser_template_argument): Don't
fold_non_dependent_expr.
diff --git a/gcc/cp/parser.c b/gcc/cp/parser.c
index 8b6dbe1..0222e90 100644
--- a/gcc/cp/parser.c
+++ b/gcc/cp/parser.c
@@ -13335,7 +13335,6 @@ cp_parser_template_argument (cp_parser* parser)
argument = cp_parser_constant_expression (parser,
/*allow_non_constant_p=*/false,
/*non_constant_p=*/NULL);
- argument = fold_non_dependent_expr (argument);
if (!maybe_type_id)
return argument;
if (!cp_parser_next_token_ends_template_argument_p (parser))
diff --git a/gcc/testsuite/g++.dg/cpp0x/constexpr-template4.C b/gcc/testsuite/g++.dg/cpp0x/constexpr-template4.C
new file mode 100644
index 0000000..7adcae8
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/constexpr-template4.C
@@ -0,0 +1,26 @@
+// PR c++/55931
+// { dg-do compile { target c++11 } }
+
+#include <type_traits>
+
+template<typename Type>
+class Test
+{
+ public:
+ constexpr Test(const Type val) : _value(val) {}
+ constexpr Type get() const {return _value;}
+ static void test()
+ {
+ static constexpr Test<int> x(42);
+ std::integral_constant<int, x.get()> i; // This is not working
+ }
+ protected:
+ Type _value;
+};
+
+int main()
+{
+ static constexpr Test<int> x(42);
+ std::integral_constant<int, x.get()> i; // This is working
+ Test<double>::test();
+}