This patch fixes PR13624, which notes a 64-bit PowerPC ELF ABI

incompatibility with how complex values are returned.  It is sufficient
to flag all complex types as direct rather than indirect.

A new test case is provided that checks correct IR generation for the
various supported flavors of _Complex.

llvm-svn: 170302
This commit is contained in:
Bill Schmidt 2012-12-17 04:20:17 +00:00
parent ccca504f3a
commit a3d121cd61
2 changed files with 130 additions and 0 deletions

View File

@ -2804,6 +2804,9 @@ PPC64_SVR4_ABIInfo::classifyReturnType(QualType RetTy) const {
if (RetTy->isVoidType()) if (RetTy->isVoidType())
return ABIArgInfo::getIgnore(); return ABIArgInfo::getIgnore();
if (RetTy->isAnyComplexType())
return ABIArgInfo::getDirect();
if (isAggregateTypeForABI(RetTy)) if (isAggregateTypeForABI(RetTy))
return ABIArgInfo::getIndirect(0); return ABIArgInfo::getIndirect(0);

View File

@ -0,0 +1,127 @@
// REQUIRES: ppc64-registered-target
// RUN: %clang_cc1 -triple powerpc64-unknown-linux-gnu -emit-llvm -o - %s | FileCheck %s
float crealf(_Complex float);
double creal(_Complex double);
long double creall(_Complex long double);
_Complex float foo_float(_Complex float x) {
return x;
}
// CHECK: define { float, float } @foo_float(float {{[%A-Za-z0-9.]+}}, float {{[%A-Za-z0-9.]+}}) nounwind {
_Complex double foo_double(_Complex double x) {
return x;
}
// CHECK: define { double, double } @foo_double(double {{[%A-Za-z0-9.]+}}, double {{[%A-Za-z0-9.]+}}) nounwind {
_Complex long double foo_long_double(_Complex long double x) {
return x;
}
// CHECK: define { ppc_fp128, ppc_fp128 } @foo_long_double(ppc_fp128 {{[%A-Za-z0-9.]+}}, ppc_fp128 {{[%A-Za-z0-9.]+}}) nounwind {
_Complex int foo_int(_Complex int x) {
return x;
}
// CHECK: define { i32, i32 } @foo_int(i32 {{[%A-Za-z0-9.]+}}, i32 {{[%A-Za-z0-9.]+}}) nounwind {
_Complex short foo_short(_Complex short x) {
return x;
}
// CHECK: define { i16, i16 } @foo_short(i16 {{[%A-Za-z0-9.]+}}, i16 {{[%A-Za-z0-9.]+}}) nounwind {
_Complex signed char foo_char(_Complex signed char x) {
return x;
}
// CHECK: define { i8, i8 } @foo_char(i8 {{[%A-Za-z0-9.]+}}, i8 {{[%A-Za-z0-9.]+}}) nounwind {
_Complex long foo_long(_Complex long x) {
return x;
}
// CHECK: define { i64, i64 } @foo_long(i64 {{[%A-Za-z0-9.]+}}, i64 {{[%A-Za-z0-9.]+}}) nounwind {
_Complex long long foo_long_long(_Complex long long x) {
return x;
}
// CHECK: define { i64, i64 } @foo_long_long(i64 {{[%A-Za-z0-9.]+}}, i64 {{[%A-Za-z0-9.]+}}) nounwind {
float bar_float(void) {
return crealf(foo_float(2.0f - 2.5fi));
}
// CHECK: define float @bar_float() nounwind {
// CHECK: [[VAR1:[%A-Za-z0-9.]+]] = call { float, float } @foo_float
// CHECK: extractvalue { float, float } [[VAR1]], 0
// CHECK: extractvalue { float, float } [[VAR1]], 1
double bar_double(void) {
return creal(foo_double(2.0 - 2.5i));
}
// CHECK: define double @bar_double() nounwind {
// CHECK: [[VAR2:[%A-Za-z0-9.]+]] = call { double, double } @foo_double
// CHECK: extractvalue { double, double } [[VAR2]], 0
// CHECK: extractvalue { double, double } [[VAR2]], 1
long double bar_long_double(void) {
return creall(foo_long_double(2.0L - 2.5Li));
}
// CHECK: define ppc_fp128 @bar_long_double() nounwind {
// CHECK: [[VAR3:[%A-Za-z0-9.]+]] = call { ppc_fp128, ppc_fp128 } @foo_long_double
// CHECK: extractvalue { ppc_fp128, ppc_fp128 } [[VAR3]], 0
// CHECK: extractvalue { ppc_fp128, ppc_fp128 } [[VAR3]], 1
int bar_int(void) {
return __real__(foo_int(2 - 3i));
}
// CHECK: define signext i32 @bar_int() nounwind {
// CHECK: [[VAR4:[%A-Za-z0-9.]+]] = call { i32, i32 } @foo_int
// CHECK: extractvalue { i32, i32 } [[VAR4]], 0
// CHECK: extractvalue { i32, i32 } [[VAR4]], 1
short bar_short(void) {
return __real__(foo_short(2 - 3i));
}
// CHECK: define signext i16 @bar_short() nounwind {
// CHECK: [[VAR5:[%A-Za-z0-9.]+]] = call { i16, i16 } @foo_short
// CHECK: extractvalue { i16, i16 } [[VAR5]], 0
// CHECK: extractvalue { i16, i16 } [[VAR5]], 1
signed char bar_char(void) {
return __real__(foo_char(2 - 3i));
}
// CHECK: define signext i8 @bar_char() nounwind {
// CHECK: [[VAR6:[%A-Za-z0-9.]+]] = call { i8, i8 } @foo_char
// CHECK: extractvalue { i8, i8 } [[VAR6]], 0
// CHECK: extractvalue { i8, i8 } [[VAR6]], 1
long bar_long(void) {
return __real__(foo_long(2L - 3Li));
}
// CHECK: define i64 @bar_long() nounwind {
// CHECK: [[VAR7:[%A-Za-z0-9.]+]] = call { i64, i64 } @foo_long
// CHECK: extractvalue { i64, i64 } [[VAR7]], 0
// CHECK: extractvalue { i64, i64 } [[VAR7]], 1
long long bar_long_long(void) {
return __real__(foo_long_long(2LL - 3LLi));
}
// CHECK: define i64 @bar_long_long() nounwind {
// CHECK: [[VAR8:[%A-Za-z0-9.]+]] = call { i64, i64 } @foo_long_long
// CHECK: extractvalue { i64, i64 } [[VAR8]], 0
// CHECK: extractvalue { i64, i64 } [[VAR8]], 1