aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--CodeGen/CGExpr.cpp36
-rw-r--r--test/CodeGen/complex.c2
2 files changed, 34 insertions, 4 deletions
diff --git a/CodeGen/CGExpr.cpp b/CodeGen/CGExpr.cpp
index 98fae6d2b3..2f9d0ea072 100644
--- a/CodeGen/CGExpr.cpp
+++ b/CodeGen/CGExpr.cpp
@@ -1070,7 +1070,23 @@ RValue CodeGenFunction::EmitMul(RValue LHS, RValue RHS, QualType ResTy) {
if (LHS.isScalar())
return RValue::get(Builder.CreateMul(LHS.getVal(), RHS.getVal(), "mul"));
- assert(0 && "FIXME: This doesn't handle complex operands yet");
+ // Otherwise, this must be a complex number.
+ llvm::Value *LHSR, *LHSI, *RHSR, *RHSI;
+
+ EmitLoadOfComplex(LHS, LHSR, LHSI);
+ EmitLoadOfComplex(RHS, RHSR, RHSI);
+
+ llvm::Value *ResRl = Builder.CreateMul(LHSR, RHSR, "mul.rl");
+ llvm::Value *ResRr = Builder.CreateMul(LHSI, RHSI, "mul.rr");
+ llvm::Value *ResR = Builder.CreateSub(ResRl, ResRr, "mul.r");
+
+ llvm::Value *ResIl = Builder.CreateMul(LHSI, RHSR, "mul.il");
+ llvm::Value *ResIr = Builder.CreateMul(LHSR, RHSI, "mul.ir");
+ llvm::Value *ResI = Builder.CreateAdd(ResIl, ResIr, "mul.i");
+
+ llvm::Value *Res = CreateTempAlloca(ConvertType(ResTy));
+ EmitStoreOfComplex(ResR, ResI, Res);
+ return RValue::getAggregate(Res);
}
RValue CodeGenFunction::EmitDiv(RValue LHS, RValue RHS, QualType ResTy) {
@@ -1231,9 +1247,23 @@ RValue CodeGenFunction::EmitBinaryCompare(const BinaryOperator *E,
}
} else {
// Struct/union/complex
- assert(0 && "Aggregate comparisons not implemented yet!");
+ llvm::Value *LHSR, *LHSI, *RHSR, *RHSI, *ResultR, *ResultI;
+ EmitLoadOfComplex(LHS, LHSR, LHSI);
+ EmitLoadOfComplex(RHS, RHSR, RHSI);
+
+ ResultR = Builder.CreateFCmp((llvm::FCmpInst::Predicate)FCmpOpc,
+ LHSR, RHSR, "cmp.r");
+ ResultI = Builder.CreateFCmp((llvm::FCmpInst::Predicate)FCmpOpc,
+ LHSI, RHSI, "cmp.i");
+ if (BinaryOperator::EQ == E->getOpcode()) {
+ Result = Builder.CreateAnd(ResultR, ResultI, "and.ri");
+ } else if (BinaryOperator::NE == E->getOpcode()) {
+ Result = Builder.CreateOr(ResultR, ResultI, "or.ri");
+ } else {
+ assert(0 && "Complex comparison other than == or != ?");
+ }
}
-
+
// ZExt result to int.
return RValue::get(Builder.CreateZExt(Result, LLVMIntTy, "cmp.ext"));
}
diff --git a/test/CodeGen/complex.c b/test/CodeGen/complex.c
index 001d148bc5..30793aaf9e 100644
--- a/test/CodeGen/complex.c
+++ b/test/CodeGen/complex.c
@@ -6,5 +6,5 @@ void main(void)
double _Complex a = 5;
double _Complex b = 42;
- return a * b == 123;
+ return a * b == b * a;
}