From 09d1369964f1f071bdede1d74c94a2d3247b28fa Mon Sep 17 00:00:00 2001 From: John McCall Date: Tue, 5 Oct 2010 22:36:42 +0000 Subject: [PATCH] When instantiating a new-expression, force a rebuild if there were default arguments in either the placement or constructor arguments. This is important if the default arguments refer to a declaration or create a temporary. llvm-svn: 115700 --- clang/lib/Sema/TreeTransform.h | 9 +++++- clang/test/CodeGenCXX/goto.cpp | 54 +++++++++++++++++++++------------- 2 files changed, 42 insertions(+), 21 deletions(-) diff --git a/clang/lib/Sema/TreeTransform.h b/clang/lib/Sema/TreeTransform.h index 135ca46958d8..5eac2cbf7cf2 100644 --- a/clang/lib/Sema/TreeTransform.h +++ b/clang/lib/Sema/TreeTransform.h @@ -5278,6 +5278,11 @@ TreeTransform::TransformCXXNewExpr(CXXNewExpr *E) { bool ArgumentChanged = false; ASTOwningVector PlacementArgs(SemaRef); for (unsigned I = 0, N = E->getNumPlacementArgs(); I != N; ++I) { + if (getDerived().DropCallArgument(E->getPlacementArg(I))) { + ArgumentChanged = true; + break; + } + ExprResult Arg = getDerived().TransformExpr(E->getPlacementArg(I)); if (Arg.isInvalid()) return ExprError(); @@ -5289,8 +5294,10 @@ TreeTransform::TransformCXXNewExpr(CXXNewExpr *E) { // transform the constructor arguments (if any). ASTOwningVector ConstructorArgs(SemaRef); for (unsigned I = 0, N = E->getNumConstructorArgs(); I != N; ++I) { - if (getDerived().DropCallArgument(E->getConstructorArg(I))) + if (getDerived().DropCallArgument(E->getConstructorArg(I))) { + ArgumentChanged = true; break; + } ExprResult Arg = getDerived().TransformExpr(E->getConstructorArg(I)); if (Arg.isInvalid()) diff --git a/clang/test/CodeGenCXX/goto.cpp b/clang/test/CodeGenCXX/goto.cpp index 2104b89b74e0..2a44d726c173 100644 --- a/clang/test/CodeGenCXX/goto.cpp +++ b/clang/test/CodeGenCXX/goto.cpp @@ -1,29 +1,43 @@ -// RUN: %clang-cc1 %s -fexceptions -emit-llvm-only +// RUN: %clang-cc1 %s -triple=x86_64-apple-darwin10 -fexceptions -emit-llvm -o - | FileCheck %s // Reduced from a crash on boost::interprocess's node_allocator_test.cpp. namespace test0 { struct A { A(); ~A(); }; struct V { V(const A &a = A()); ~V(); }; - template int vector_test() - { - A process_name; - try { - A segment; + // CHECK: define linkonce_odr i32 @_ZN5test04testILi0EEEii + template int test(int x) { + // CHECK: [[RET:%.*]] = alloca i32 + // CHECK-NEXT: [[X:%.*]] = alloca i32 + // CHECK-NEXT: [[Y:%.*]] = alloca [[A:%.*]], + // CHECK-NEXT: [[Z:%.*]] = alloca [[A]] + // CHECK-NEXT: [[EXN:%.*]] = alloca i8* + // CHECK-NEXT: alloca i32 + // CHECK-NEXT: [[V:%.*]] = alloca [[V:%.*]]*, + // CHECK-NEXT: [[TMP:%.*]] = alloca [[A]] + // CHECK-NEXT: [[CLEANUPACTIVE:%.*]] = alloca i1 + // CHECK: store i1 true, i1* [[CLEANUPACTIVE]] + // CHECK: call void @_ZN5test01AC1Ev([[A]]* [[Y]]) + // CHECK-NEXT: invoke void @_ZN5test01AC1Ev([[A]]* [[Z]]) + // CHECK: [[NEW:%.*]] = invoke noalias i8* @_Znwm(i64 1) + // CHECK: [[NEWCAST:%.*]] = bitcast i8* [[NEW]] to [[V]]* + // CHECK-NEXT: invoke void @_ZN5test01AC1Ev([[A]]* [[TMP]]) + // CHECK: invoke void @_ZN5test01VC1ERKNS_1AE([[V]]* [[NEWCAST]], [[A]]* [[TMP]]) + // CHECK: store i1 false, i1* [[CLEANUPACTIVE]] + // CHECK-NEXT: invoke void @_ZN5test01AD1Ev([[A]]* [[TMP]]) + A y; + try { + A z; + V *v = new V(); - V *stdvector = new V(); + if (x) return 1; + } catch (int ex) { + return 1; + } + return 0; + } - int x = 5, y = 7; - if(x == y) return 1; - } - catch(int ex){ - return 1; - } - return 0; -} - -int main () -{ - return vector_test<0>(); -} + int test() { + return test<0>(5); + } }