| //==- llvm/tools/gollvm/unittests/BackendCore/BackendCABIOracleTests.cpp -==// |
| // |
| // Copyright 2018 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "TestUtils.h" |
| #include "go-llvm-cabi-oracle.h" |
| #include "go-llvm-backend.h" |
| #include "llvm/IR/Constants.h" |
| #include "llvm/IR/Function.h" |
| #include "gtest/gtest.h" |
| |
| using namespace llvm; |
| using namespace goBackendUnitTests; |
| |
| namespace { |
| |
| TEST(BackendCABIOracleTests, Basic) { |
| LLVMContext C; |
| std::unique_ptr<Llvm_backend> bep(new Llvm_backend(C, nullptr, nullptr, 0)); |
| Llvm_backend *be = bep.get(); |
| |
| Btype *bi8t = be->integer_type(false, 8); |
| Btype *bu8t = be->integer_type(true, 8); |
| Btype *bf32t = be->float_type(32); |
| Btype *bf64t = be->float_type(64); |
| Btype *st0 = mkBackendStruct(be, nullptr); |
| Btype *st1 = mkBackendStruct(be, bi8t, "a", bu8t, "b", bf32t, "c", nullptr); |
| Btype *st2 = mkBackendStruct(be, bf64t, "f1", bf64t, "f2", nullptr); |
| |
| { |
| BFunctionType *befty1 = mkFuncTyp(be, |
| L_PARM, bi8t, |
| L_PARM, bf32t, |
| L_PARM, st0, |
| L_PARM, st1, |
| L_RES, st2, |
| L_END); |
| CABIOracle cab(befty1, be->typeManager()); |
| const char *exp = R"RAW_RESULT( |
| Return: Direct { { double, double } } sigOffset: -1 |
| Param 1: Direct AttrNest { i8* } sigOffset: 0 |
| Param 2: Direct AttrSext { i8 } sigOffset: 1 |
| Param 3: Direct { float } sigOffset: 2 |
| Param 4: Ignore { void } sigOffset: -1 |
| Param 5: Direct { i64 } sigOffset: 3 |
| )RAW_RESULT"; |
| std::string reason; |
| bool equal = difftokens(exp, cab.toString(), reason); |
| EXPECT_EQ("pass", equal ? "pass" : reason); |
| EXPECT_EQ(repr(cab.getFunctionTypeForABI()), |
| "{ double, double } (i8*, i8, float, i64)"); |
| } |
| } |
| |
| TEST(BackendCABIOracleTests, Extended) { |
| LLVMContext C; |
| std::unique_ptr<Llvm_backend> bep(new Llvm_backend(C, nullptr, nullptr, 0)); |
| Llvm_backend *be = bep.get(); |
| |
| Btype *bi8t = be->integer_type(false, 8); |
| Btype *bu8t = be->integer_type(true, 8); |
| Btype *bu64t = be->integer_type(true, 64); |
| Btype *bu32t = be->integer_type(true, 32); |
| Btype *bi16t = be->integer_type(false, 16); |
| Btype *bf32t = be->float_type(32); |
| Btype *bf64t = be->float_type(64); |
| Btype *bpu64t = be->pointer_type(bu64t); |
| Btype *bpf64t = be->pointer_type(bf64t); |
| Btype *st0 = mkBackendStruct(be, nullptr); |
| Btype *st1 = mkBackendStruct(be, bi8t, "a", bu8t, "b", bf32t, "c", nullptr); |
| Btype *st2 = mkBackendStruct(be, bf64t, "f1", bf64t, "f2", nullptr); |
| Btype *st3 = mkBackendStruct(be, st2, "f1", bi8t, "f2", nullptr); |
| Btype *st4 = mkBackendStruct(be, bf32t, "f1", bf32t, "f2", nullptr); |
| Btype *st5 = mkBackendStruct(be, bf32t, "f1", nullptr); |
| Btype *st6 = mkBackendStruct(be, bf32t, "f1", bi8t, "a", bu8t, "b", |
| bu64t, "c", nullptr); |
| Btype *st7 = mkBackendStruct(be, bf32t, "f1", bu32t, "f2", nullptr); |
| Btype *st8 = mkBackendStruct(be, bi8t, "f1", bi16t, "f2", st7, "f3", nullptr); |
| Btype *stii = mkBackendStruct(be, bu64t, "a", bu64t, "b", nullptr); |
| Btype *stip = mkBackendStruct(be, bu64t, "a", bpu64t, "b", nullptr); |
| Btype *stpi = mkBackendStruct(be, bpu64t, "a", bu64t, "b", nullptr); |
| Btype *stpp = mkBackendStruct(be, bpu64t, "a", bpu64t, "b", nullptr); |
| Btype *at0 = be->array_type(bu32t, mkInt64Const(be, int64_t(0))); |
| Btype *at1 = be->array_type(bu32t, mkInt64Const(be, int64_t(1))); |
| Btype *at2 = be->array_type(bu32t, mkInt64Const(be, int64_t(3))); |
| Btype *at3 = be->array_type(bu8t, mkInt64Const(be, int64_t(16))); |
| |
| struct FcnItem { |
| FcnItem(const std::vector<Btype*> &r, |
| const std::vector<Btype*> &p, |
| const char *d, const char *t) |
| : results(r), parms(p), expDump(d), expTyp(t) { } |
| std::vector<Btype*> results; |
| std::vector<Btype*> parms; |
| const char *expDump; |
| const char *expTyp; |
| }; |
| |
| Btype *nt = nullptr; |
| std::vector<FcnItem> items = { |
| |
| // 1 |
| FcnItem( { }, { }, |
| "Return: Ignore { void } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0", |
| "void (i8*)"), |
| |
| // 2 |
| FcnItem( { bi8t }, { }, |
| "Return: Direct AttrSext { i8 } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0", |
| "i8 (i8*)"), |
| |
| // 3 |
| FcnItem( { }, { bi8t }, |
| "Return: Ignore { void } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct AttrSext { i8 } sigOffset: 1", |
| "void (i8*, i8)"), |
| |
| // 4 |
| FcnItem( { }, { st5, bpf64t }, |
| "Return: Ignore { void } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { float } sigOffset: 1 " |
| "Param 3: Direct { double* } sigOffset: 2", |
| "void (i8*, float, double*)"), |
| |
| // 5 |
| FcnItem({ bi8t, bf64t }, { bi8t, bu8t, st0 }, |
| "Return: Direct { { i8, double } } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct AttrSext { i8 } sigOffset: 1 " |
| "Param 3: Direct AttrZext { i8 } sigOffset: 2 " |
| "Param 4: Ignore { void } sigOffset: -1", |
| "{ i8, double } (i8*, i8, i8)"), |
| |
| // 6 |
| FcnItem({ st2 }, { st2, st0, st4, st1 }, |
| "Return: Direct { { double, double } } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { double, double } sigOffset: 1 " |
| "Param 3: Ignore { void } sigOffset: -1 " |
| "Param 4: Direct { <2 x float> } sigOffset: 3 " |
| "Param 5: Direct { i64 } sigOffset: 4 ", |
| "{ double, double } (i8*, double, double, <2 x float>, i64)"), |
| |
| // 7 |
| FcnItem({ st3 }, { st3, st0, bu8t }, |
| "Return: Indirect AttrStructReturn { { { double, double }, i8 }* } sigOffset: 0 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 1 " |
| "Param 2: Indirect AttrByVal { { { double, double }, i8 }* } sigOffset: 2 " |
| "Param 3: Ignore { void } sigOffset: -1 " |
| "Param 4: Direct AttrZext { i8 } sigOffset: 3 ", |
| "void ({ { double, double }, i8 }*, i8*, " |
| "{ { double, double }, i8 }*, i8)"), |
| |
| // 8 |
| FcnItem( { st6 }, { st6, st6 }, |
| "Return: Direct { { i64, i64 } } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { i64, i64 } sigOffset: 1 " |
| "Param 3: Direct { i64, i64 } sigOffset: 3", |
| "{ i64, i64 } (i8*, i64, i64, i64, i64)"), |
| |
| // 9 |
| FcnItem( { st8 }, { st8 }, |
| "Return: Direct { { i64, i32 } } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { i64, i32 } sigOffset: 1", |
| "{ i64, i32 } (i8*, i64, i32)"), |
| |
| // 10 |
| FcnItem( { at0 }, { at1 }, |
| "Return: Ignore { void } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { i32 } sigOffset: 1", |
| "void (i8*, i32)"), |
| |
| // 11 |
| FcnItem( { at2 }, { at3 }, |
| "Return: Direct { { i64, i32 } } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { i64, i64 } sigOffset: 1", |
| "{ i64, i32 } (i8*, i64, i64)"), |
| |
| // 12 |
| // Make sure pointerness is preserved. |
| FcnItem( { stip }, { stii, stpp, stpi }, |
| "Return: Direct { { i64, i8* } } sigOffset: -1 " |
| "Param 1: Direct AttrNest { i8* } sigOffset: 0 " |
| "Param 2: Direct { i64, i64 } sigOffset: 1 " |
| "Param 3: Direct { i8*, i8* } sigOffset: 3 " |
| "Param 4: Direct { i8*, i64 } sigOffset: 5", |
| "{ i64, i8* } (i8*, i64, i64, i8*, i8*, i8*, i64)"), |
| }; |
| |
| unsigned count = 1; |
| for (auto &item : items) { |
| std::vector<Backend::Btyped_identifier> results; |
| std::vector<Backend::Btyped_identifier> params; |
| for (auto &r : item.results) |
| results.push_back(mkid(r)); |
| for (auto &p : item.parms) |
| params.push_back(mkid(p)); |
| Btype *rt = nullptr; |
| if (results.size() > 1) |
| rt = be->struct_type(results); |
| Btype *t = be->function_type(mkid(nt), params, results, rt, Location()); |
| BFunctionType *bft = t->castToBFunctionType(); |
| CABIOracle cab(bft, be->typeManager()); |
| |
| { std::string reason; |
| bool equal = difftokens(item.expDump, cab.toString(), reason); |
| EXPECT_EQ("pass", equal ? "pass" : reason); |
| if (!equal) { |
| std::cerr << "count: " << count << "\n"; |
| std::cerr << "exp:\n" << item.expDump << "\n"; |
| std::cerr << "act:\n" << cab.toString() << "\n"; |
| } |
| } |
| { std::string reason; |
| std::string result(repr(cab.getFunctionTypeForABI())); |
| bool equal = difftokens(item.expTyp, result, reason); |
| EXPECT_EQ("pass", equal ? "pass" : reason); |
| if (!equal) { |
| std::cerr << "count: " << count << "\n"; |
| std::cerr << "exp:\n" << item.expTyp << "\n"; |
| std::cerr << "act:\n" << result << "\n"; |
| } |
| } |
| count++; |
| } |
| } |
| |
| TEST(BackendCABIOracleTests, RecursiveCall1) { |
| FcnTestHarness h; |
| Llvm_backend *be = h.be(); |
| |
| // type s1 struct { |
| // f1, f2 float32 |
| // i1, i2, i3 int16 |
| // } |
| // type s2 struct { |
| // k float64 |
| // f1, f2 float32 |
| // } |
| // type s3 struct { |
| // f1, f2 s1 |
| // } |
| // type s4 struct { |
| // } |
| // func foo(x s1, y s2, z s4, sm1 uint8, sm2 int16, w s3) s2 { |
| // if (sm1 == 0) { |
| // return y |
| // } |
| // return foo(x, y, z, sm1-1, sm2, s3) |
| // } |
| // |
| |
| // Create struct types |
| Btype *bf32t = be->float_type(32); |
| Btype *bf64t = be->float_type(64); |
| Btype *bi16t = be->integer_type(false, 16); |
| Btype *bi8t = be->integer_type(false, 8); |
| Btype *bu8t = be->integer_type(true, 8); |
| Btype *s1 = mkBackendStruct(be, bf32t, "f1", bf32t, "f2", |
| bi16t, "i1", bi16t, "i2", bi16t, "i3", nullptr); |
| Btype *s2 = mkBackendStruct(be, bf64t, "k", bf32t, "f1", bf32t, "f2", |
| nullptr); |
| Btype *s3 = mkBackendStruct(be, s1, "f1", s2, "f2", nullptr); |
| Btype *s4 = mkBackendStruct(be, nullptr); |
| |
| // Create function type |
| BFunctionType *befty1 = mkFuncTyp(be, |
| L_PARM, s1, |
| L_PARM, s2, |
| L_PARM, s4, |
| L_PARM, bu8t, |
| L_PARM, bi8t, |
| L_PARM, s3, |
| L_RES, s2, |
| L_END); |
| Bfunction *func = h.mkFunction("foo", befty1); |
| |
| // sm1 == 0 |
| Bvariable *p3 = func->getNthParamVar(3); |
| Location loc; |
| Bexpression *vex = be->var_expression(p3, loc); |
| Bexpression *c0 = be->convert_expression(bu8t, mkInt32Const(be, 0), loc); |
| Bexpression *eq = be->binary_expression(OPERATOR_EQEQ, vex, c0, loc); |
| |
| // call |
| Bexpression *fn = be->function_code_expression(func, loc); |
| std::vector<Bexpression *> args; |
| Bvariable *p0 = func->getNthParamVar(0); |
| args.push_back(be->var_expression(p0, loc)); |
| |
| Bvariable *p1 = func->getNthParamVar(1); |
| args.push_back(be->var_expression(p1, loc)); |
| |
| Bvariable *p2 = func->getNthParamVar(2); |
| args.push_back(be->var_expression(p2, loc)); |
| |
| Bvariable *p3x = func->getNthParamVar(3); |
| Bexpression *vex3 = be->var_expression(p3x, loc); |
| Bexpression *c1 = be->convert_expression(bu8t, mkInt32Const(be, 1), loc); |
| Bexpression *minus = be->binary_expression(OPERATOR_MINUS, vex3, c1, loc); |
| args.push_back(minus); |
| |
| Bvariable *p4 = func->getNthParamVar(4); |
| args.push_back(be->var_expression(p4, loc)); |
| |
| Bvariable *p5 = func->getNthParamVar(5); |
| args.push_back(be->var_expression(p5, loc)); |
| Bexpression *call = be->call_expression(func, fn, args, nullptr, h.loc()); |
| |
| // return y |
| std::vector<Bexpression *> rvals1; |
| rvals1.push_back(be->var_expression(p1, loc)); |
| Bstatement *rst1 = h.mkReturn(rvals1, FcnTestHarness::NoAppend); |
| |
| // return call |
| std::vector<Bexpression *> rvals2; |
| rvals2.push_back(call); |
| Bstatement *rst2 = h.mkReturn(rvals2, FcnTestHarness::NoAppend); |
| |
| const char *exp = R"RAW_RESULT( |
| %p3.ld.0 = load i8, i8* %p3.addr |
| %sub.0 = sub i8 %p3.ld.0, 1 |
| %p4.ld.0 = load i8, i8* %p4.addr |
| %cast.1 = bitcast { float, float, i16, i16, i16 }* %p0.addr to { <2 x float>, i48 }* |
| %field0.0 = getelementptr inbounds { <2 x float>, i48 }, { <2 x float>, i48 }* %cast.1, i32 0, i32 0 |
| %ld.1 = load <2 x float>, <2 x float>* %field0.0 |
| %field1.0 = getelementptr inbounds { <2 x float>, i48 }, { <2 x float>, i48 }* %cast.1, i32 0, i32 1 |
| %ld.2 = load i48, i48* %field1.0 |
| %cast.2 = bitcast { double, float, float }* %p1.addr to { double, <2 x float> }* |
| %field0.1 = getelementptr inbounds { double, <2 x float> }, { double, <2 x float> }* %cast.2, i32 0, i32 0 |
| %ld.3 = load double, double* %field0.1 |
| %field1.1 = getelementptr inbounds { double, <2 x float> }, { double, <2 x float> }* %cast.2, i32 0, i32 1 |
| %ld.4 = load <2 x float>, <2 x float>* %field1.1 |
| %call.0 = call addrspace(0) { double, <2 x float> } @foo(i8* nest undef, <2 x float> %ld.1, i48 %ld.2, double %ld.3, <2 x float> %ld.4, i8 zeroext %sub.0, i8 signext %p4.ld.0, { { float, float, i16, i16, i16 }, { double, float, float } }* byval %p5) |
| %cast.3 = bitcast { double, float, float }* %sret.actual.0 to { double, <2 x float> }* |
| store { double, <2 x float> } %call.0, { double, <2 x float> }* %cast.3 |
| %cast.4 = bitcast { double, float, float }* %sret.actual.0 to { double, <2 x float> }* |
| %ld.5 = load { double, <2 x float> }, { double, <2 x float> }* %cast.4 |
| ret { double, <2 x float> } %ld.5 |
| )RAW_RESULT"; |
| |
| bool isOK = h.expectStmt(rst2, exp); |
| EXPECT_TRUE(isOK && "Statement does not have expected contents"); |
| |
| // if statement |
| h.mkIf(eq, rst1, rst2); |
| |
| bool broken = h.finish(PreserveDebugInfo); |
| EXPECT_FALSE(broken && "Module failed to verify."); |
| |
| } |
| |
| TEST(BackendCABIOracleTests, PassAndReturnArrays) { |
| FcnTestHarness h; |
| Llvm_backend *be = h.be(); |
| |
| Btype *bf32t = be->float_type(32); |
| Btype *bf64t = be->float_type(64); |
| Btype *at2f = be->array_type(bf32t, mkInt64Const(be, int64_t(2))); |
| Btype *at3d = be->array_type(bf64t, mkInt64Const(be, int64_t(3))); |
| |
| // func foo(fp [2]float32) [3]float64 |
| BFunctionType *befty1 = mkFuncTyp(be, |
| L_PARM, at2f, |
| L_RES, at3d, |
| L_END); |
| Bfunction *func = h.mkFunction("foo", befty1); |
| |
| // foo(fp) |
| Location loc; |
| Bvariable *p0 = func->getNthParamVar(0); |
| Bexpression *vex = be->var_expression(p0, loc); |
| Bexpression *fn = be->function_code_expression(func, loc); |
| std::vector<Bexpression *> args; |
| args.push_back(vex); |
| Bexpression *call = be->call_expression(func, fn, args, nullptr, h.loc()); |
| |
| // return foo(fp) |
| std::vector<Bexpression *> rvals; |
| rvals.push_back(call); |
| h.mkReturn(rvals); |
| |
| const char *exp = R"RAW_RESULT( |
| %cast.0 = bitcast [2 x float]* %p0.addr to <2 x float>* |
| %ld.0 = load <2 x float>, <2 x float>* %cast.0 |
| call addrspace(0) void @foo([3 x double]* sret %sret.actual.0, i8* nest undef, <2 x float> %ld.0) |
| %cast.1 = bitcast [3 x double]* %sret.formal.0 to i8* |
| %cast.2 = bitcast [3 x double]* %sret.actual.0 to i8* |
| call addrspace(0) void @llvm.memcpy.p0i8.p0i8.i64(i8* align 8 %cast.1, i8* align 8 %cast.2, i64 24, i1 false) |
| ret void |
| )RAW_RESULT"; |
| |
| bool isOK = h.expectBlock(exp); |
| EXPECT_TRUE(isOK && "Block does not have expected contents"); |
| |
| bool broken = h.finish(PreserveDebugInfo); |
| EXPECT_FALSE(broken && "Module failed to verify."); |
| } |
| |
| TEST(BackendCABIOracleTests, EmptyStructParamsAndReturns) { |
| FcnTestHarness h; |
| Llvm_backend *be = h.be(); |
| |
| Btype *bi32t = be->integer_type(false, 32); |
| Btype *set = mkBackendStruct(be, nullptr); // struct with no fields |
| |
| // func foo(f1 empty, f2 empty, f3 int32, f4 empty) empty |
| BFunctionType *befty1 = mkFuncTyp(be, |
| L_RES, set, |
| L_PARM, set, |
| L_PARM, set, |
| L_PARM, bi32t, |
| L_PARM, set, |
| L_PARM, set, |
| L_END); |
| Bfunction *func = h.mkFunction("foo", befty1); |
| |
| // foo(f1, f1, 4, f1, f1) |
| Location loc; |
| Bexpression *fn = be->function_code_expression(func, loc); |
| Bvariable *p0 = func->getNthParamVar(0); |
| std::vector<Bexpression *> args; |
| args.push_back(be->var_expression(p0, loc)); |
| args.push_back(be->var_expression(p0, loc)); |
| args.push_back(mkInt32Const(be, 4)); |
| args.push_back(be->var_expression(p0, loc)); |
| args.push_back(be->var_expression(p0, loc)); |
| Bexpression *call = be->call_expression(func, fn, args, nullptr, h.loc()); |
| |
| // return the call above |
| std::vector<Bexpression *> rvals; |
| rvals.push_back(call); |
| h.mkReturn(rvals); |
| |
| const char *exp = R"RAW_RESULT( |
| call addrspace(0) void @foo(i8* nest undef, i32 4) |
| ret void |
| )RAW_RESULT"; |
| |
| bool isOK = h.expectBlock(exp); |
| EXPECT_TRUE(isOK && "Block does not have expected contents"); |
| |
| bool broken = h.finish(PreserveDebugInfo); |
| EXPECT_FALSE(broken && "Module failed to verify."); |
| } |
| |
| TEST(BackendCABIOracleTests, CallBuiltinFunction) { |
| FcnTestHarness h("foo"); |
| Llvm_backend *be = h.be(); |
| Bfunction *func = h.func(); |
| |
| Bfunction *tfunc = be->lookup_builtin("__builtin_trap"); |
| |
| // trap() |
| Location loc; |
| Bexpression *fn = be->function_code_expression(tfunc, loc); |
| std::vector<Bexpression *> args; |
| h.mkExprStmt(be->call_expression(func, fn, args, nullptr, h.loc())); |
| |
| const char *exp = R"RAW_RESULT( |
| call addrspace(0) void @llvm.trap() |
| )RAW_RESULT"; |
| |
| bool isOK = h.expectBlock(exp); |
| EXPECT_TRUE(isOK && "Block does not have expected contents"); |
| |
| bool broken = h.finish(PreserveDebugInfo); |
| EXPECT_FALSE(broken && "Module failed to verify."); |
| } |
| |
| TEST(BackendCABIOracleTests, PassAndReturnComplex) { |
| FcnTestHarness h; |
| Llvm_backend *be = h.be(); |
| |
| Btype *bc64t = be->complex_type(64); |
| Btype *bc128t = be->complex_type(128); |
| |
| // func foo(x complex64, y complex128) complex64 |
| BFunctionType *befty1 = mkFuncTyp(be, |
| L_PARM, bc64t, |
| L_PARM, bc128t, |
| L_RES, bc64t, |
| L_END); |
| Bfunction *func = h.mkFunction("foo", befty1); |
| |
| // z = foo(x, y) |
| Location loc; |
| Bvariable *x = func->getNthParamVar(0); |
| Bvariable *y = func->getNthParamVar(1); |
| Bexpression *xvex = be->var_expression(x, loc); |
| Bexpression *yvex = be->var_expression(y, loc); |
| Bexpression *fn1 = be->function_code_expression(func, loc); |
| std::vector<Bexpression *> args1 = {xvex, yvex}; |
| Bexpression *call1 = be->call_expression(func, fn1, args1, nullptr, h.loc()); |
| h.mkLocal("z", bc64t, call1); |
| |
| // Call with constant args |
| // foo(1+2i, 3+4i) |
| mpc_t mpc_val1, mpc_val2; |
| mpc_init2(mpc_val1, 256); |
| mpc_set_d_d(mpc_val1, 1.0, 2.0, GMP_RNDN); |
| mpc_init2(mpc_val2, 256); |
| mpc_set_d_d(mpc_val2, 3.0, 4.0, GMP_RNDN); |
| Bexpression *ccon1 = be->complex_constant_expression(bc64t, mpc_val1); |
| Bexpression *ccon2 = be->complex_constant_expression(bc128t, mpc_val2); |
| mpc_clear(mpc_val1); |
| mpc_clear(mpc_val2); |
| Bexpression *fn2 = be->function_code_expression(func, loc); |
| std::vector<Bexpression *> args2 = {ccon1, ccon2}; |
| Bexpression *call2 = be->call_expression(func, fn2, args2, nullptr, h.loc()); |
| |
| // return the call expr above |
| std::vector<Bexpression *> rvals = {call2}; |
| h.mkReturn(rvals); |
| |
| const char *exp = R"RAW_RESULT( |
| %cast.0 = bitcast { float, float }* %p0.addr to <2 x float>* |
| %ld.0 = load <2 x float>, <2 x float>* %cast.0 |
| %field0.0 = getelementptr inbounds { double, double }, { double, double }* %p1.addr, i32 0, i32 0 |
| %ld.1 = load double, double* %field0.0 |
| %field1.0 = getelementptr inbounds { double, double }, { double, double }* %p1.addr, i32 0, i32 1 |
| %ld.2 = load double, double* %field1.0 |
| %call.0 = call addrspace(0) <2 x float> @foo(i8* nest undef, <2 x float> %ld.0, double %ld.1, double %ld.2) |
| %cast.2 = bitcast { float, float }* %sret.actual.0 to <2 x float>* |
| store <2 x float> %call.0, <2 x float>* %cast.2 |
| %cast.3 = bitcast { float, float }* %z to i8* |
| %cast.4 = bitcast { float, float }* %sret.actual.0 to i8* |
| call addrspace(0) void @llvm.memcpy.p0i8.p0i8.i64(i8* align 4 %cast.3, i8* align 4 %cast.4, i64 8, i1 false) |
| %ld.3 = load <2 x float>, <2 x float>* bitcast ({ float, float }* @const.0 to <2 x float>*) |
| %ld.4 = load double, double* getelementptr inbounds ({ double, double }, { double, double }* @const.1, i32 0, i32 0) |
| %ld.5 = load double, double* getelementptr inbounds ({ double, double }, { double, double }* @const.1, i32 0, i32 1) |
| %call.1 = call addrspace(0) <2 x float> @foo(i8* nest undef, <2 x float> %ld.3, double %ld.4, double %ld.5) |
| %cast.7 = bitcast { float, float }* %sret.actual.1 to <2 x float>* |
| store <2 x float> %call.1, <2 x float>* %cast.7 |
| %cast.8 = bitcast { float, float }* %sret.actual.1 to <2 x float>* |
| %ld.6 = load <2 x float>, <2 x float>* %cast.8 |
| ret <2 x float> %ld.6 |
| )RAW_RESULT"; |
| |
| bool isOK = h.expectBlock(exp); |
| EXPECT_TRUE(isOK && "Block does not have expected contents"); |
| |
| bool broken = h.finish(PreserveDebugInfo); |
| EXPECT_FALSE(broken && "Module failed to verify."); |
| } |
| |
| } |