From 7cb17890114fa63ecdc440e1e2376726cc8fce19 Mon Sep 17 00:00:00 2001 From: Alexey Bataev Date: Thu, 18 Dec 2014 06:54:53 +0000 Subject: [PATCH] Fix for PR21915: assert on multidimensional VLA in function arguments. Fixed assertion on type checking for arguments and parameters on function call if arguments are pointers to VLA Differential Revision: http://reviews.llvm.org/D6655 llvm-svn: 224504 --- clang/lib/CodeGen/CodeGenFunction.h | 29 +++++++---------------------- clang/test/CodeGen/vlt_to_pointer.c | 30 ++++++++++++++++++++++++++++++ clang/test/CodeGenCXX/vlt_to_reference.cpp | 22 ++++++++++++++++++++++ 3 files changed, 59 insertions(+), 22 deletions(-) create mode 100644 clang/test/CodeGen/vlt_to_pointer.c create mode 100644 clang/test/CodeGenCXX/vlt_to_reference.cpp diff --git a/clang/lib/CodeGen/CodeGenFunction.h b/clang/lib/CodeGen/CodeGenFunction.h index 2e1e400..e2fd80e 100644 --- a/clang/lib/CodeGen/CodeGenFunction.h +++ b/clang/lib/CodeGen/CodeGenFunction.h @@ -2746,28 +2746,13 @@ public: E = CallArgTypeInfo->param_type_end(); I != E; ++I, ++Arg) { assert(Arg != ArgEnd && "Running over edge of argument list!"); -#ifndef NDEBUG - QualType ArgType = *I; - QualType ActualArgType = Arg->getType(); - if (ArgType->isPointerType() && ActualArgType->isPointerType()) { - QualType ActualBaseType = - ActualArgType->getAs()->getPointeeType(); - QualType ArgBaseType = - ArgType->getAs()->getPointeeType(); - if (ArgBaseType->isVariableArrayType()) { - if (const VariableArrayType *VAT = - getContext().getAsVariableArrayType(ActualBaseType)) { - if (!VAT->getSizeExpr()) - ActualArgType = ArgType; - } - } - } - assert(getContext() - .getCanonicalType(ArgType.getNonReferenceType()) - .getTypePtr() == - getContext().getCanonicalType(ActualArgType).getTypePtr() && - "type mismatch in call argument!"); -#endif + assert( + ((*I)->isVariablyModifiedType() || + getContext() + .getCanonicalType((*I).getNonReferenceType()) + .getTypePtr() == + getContext().getCanonicalType(Arg->getType()).getTypePtr()) && + "type mismatch in call argument!"); ArgTypes.push_back(*I); } } diff --git a/clang/test/CodeGen/vlt_to_pointer.c b/clang/test/CodeGen/vlt_to_pointer.c new file mode 100644 index 0000000..22c620a --- /dev/null +++ b/clang/test/CodeGen/vlt_to_pointer.c @@ -0,0 +1,30 @@ +// RUN: %clang_cc1 %s -emit-llvm -o - | FileCheck %s + +int c[1][3*2]; +// CHECK: @{{.+}} = {{.*}} global [1 x [6 x {{i[0-9]+}}]] zeroinitializer + +// CHECK-LABEL: @f +int f(int * const m, int (**v)[*m * 2]) +{ + return &(c[0][*m]) == &((*v)[0][*m]); + // CHECK: icmp + // CHECK: ret i{{[0-9]+}} +} + +// CHECK-LABEL: @test +int test(int n, int (*(*fn)(void))[n]) { + return (*fn())[0]; +} + +// CHECK-LABEL: @main +int main() +{ + int m = 3; + int (*d)[3*2] = c; + int (*fn[m])(void); + return f(&m, &d) + test(m, &fn); + + // CHECK: call {{.+}} @f( + // CHECK: ret i{{[0-9]+}} +} + diff --git a/clang/test/CodeGenCXX/vlt_to_reference.cpp b/clang/test/CodeGenCXX/vlt_to_reference.cpp new file mode 100644 index 0000000..49d7f1a --- /dev/null +++ b/clang/test/CodeGenCXX/vlt_to_reference.cpp @@ -0,0 +1,22 @@ +// RUN: %clang_cc1 %s -emit-llvm -o - | FileCheck %s + +// CHECK-LABEL: @main + +struct dyn_array { + int size; + int data[]; +}; + +int foo(dyn_array **&d) { + return (*d)->data[1]; +} + +int main() +{ + dyn_array **d; + return foo(d); + + // CHECK: call {{.+}} @{{.+}}foo{{.+}}( + // CHECK: ret i{{[0-9]+}} +} + -- 2.7.4