diff options
| author | jason <jason@138bc75d-0d04-0410-961f-82ee72b054a4> | 2006-08-17 00:51:37 +0000 |
|---|---|---|
| committer | jason <jason@138bc75d-0d04-0410-961f-82ee72b054a4> | 2006-08-17 00:51:37 +0000 |
| commit | f98a9d6fc2fe2cdc5773a7a84aa1f0ea88912d73 (patch) | |
| tree | 329d90ea87976a7bc7e05d9b522879373892aaaf | |
| parent | a1fa70081973a9068c55d73316b11f6a63b588c3 (diff) | |
| download | ppe42-gcc-f98a9d6fc2fe2cdc5773a7a84aa1f0ea88912d73.tar.gz ppe42-gcc-f98a9d6fc2fe2cdc5773a7a84aa1f0ea88912d73.zip | |
PR c++/28385
* pt.c (tsubst) [TEMPLATE_TYPE_PARM]: Ignore quals from template
if arg is a function.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@116203 138bc75d-0d04-0410-961f-82ee72b054a4
| -rw-r--r-- | gcc/cp/ChangeLog | 6 | ||||
| -rw-r--r-- | gcc/cp/pt.c | 14 | ||||
| -rw-r--r-- | gcc/testsuite/g++.dg/template/const1.C | 30 |
3 files changed, 48 insertions, 2 deletions
diff --git a/gcc/cp/ChangeLog b/gcc/cp/ChangeLog index 7b08dd72033..f69c4ff58f6 100644 --- a/gcc/cp/ChangeLog +++ b/gcc/cp/ChangeLog @@ -1,3 +1,9 @@ +2006-08-16 Jason Merrill <jason@redhat.com> + + PR c++/28385 + * pt.c (tsubst) [TEMPLATE_TYPE_PARM]: Ignore quals from template + if arg is a function. + 2006-08-16 Volker Reichelt <reichelt@igpm.rwth-aachen.de> PR c++/28593 diff --git a/gcc/cp/pt.c b/gcc/cp/pt.c index 4c2b45a5a69..27b6d111a26 100644 --- a/gcc/cp/pt.c +++ b/gcc/cp/pt.c @@ -7238,10 +7238,20 @@ tsubst (tree t, tree args, tsubst_flags_t complain, tree in_decl) { if (TREE_CODE (t) == TEMPLATE_TYPE_PARM) { + int quals; gcc_assert (TYPE_P (arg)); + + /* cv-quals from the template are discarded when + substituting in a function or reference type. */ + if (TREE_CODE (arg) == FUNCTION_TYPE + || TREE_CODE (arg) == METHOD_TYPE + || TREE_CODE (arg) == REFERENCE_TYPE) + quals = cp_type_quals (arg); + else + quals = cp_type_quals (arg) | cp_type_quals (t); + return cp_build_qualified_type_real - (arg, cp_type_quals (arg) | cp_type_quals (t), - complain | tf_ignore_bad_quals); + (arg, quals, complain | tf_ignore_bad_quals); } else if (TREE_CODE (t) == BOUND_TEMPLATE_TEMPLATE_PARM) { diff --git a/gcc/testsuite/g++.dg/template/const1.C b/gcc/testsuite/g++.dg/template/const1.C new file mode 100644 index 00000000000..629c4d4a916 --- /dev/null +++ b/gcc/testsuite/g++.dg/template/const1.C @@ -0,0 +1,30 @@ +// PR c++/28385 +// instantiating op() with void()() was making the compiler think that 'fcn' +// was const, so it could eliminate the call. + +// { dg-do run } + +extern "C" void abort (void); + +int barcnt = 0; + +class Foo { + public: + template<typename T> + void operator()(const T& fcn) { + fcn(); + } +}; + +void bar() { + barcnt++; +} + +int main() { + Foo myFoo; + myFoo(bar); + myFoo(&bar); + if (barcnt != 2) + abort (); + return 0; +} |

