//===-- abi-arm.cpp ---------------------------------------------------===// // // LDC – the LLVM D compiler // // This file is distributed under the BSD-style LDC license. See the LICENSE // file for details. // //===----------------------------------------------------------------------===// /* ARM ABI based on AAPCS (Procedure Call Standard for the ARM Architecture) http://infocenter.arm.com/help/topic/com.arm.doc.ihi0042f/IHI0042F_aapcs.pdf */ #include "gen/abi-arm.h" #include "dmd/identifier.h" #include "dmd/ldcbindings.h" #include "gen/abi.h" #include "gen/abi-generic.h" #include "llvm/Target/TargetMachine.h" struct ArmTargetABI : TargetABI { HFAToArray hfaToArray; CompositeToArray32 compositeToArray32; CompositeToArray64 compositeToArray64; IntegerRewrite integerRewrite; bool returnInArg(TypeFunction *tf, bool) override { // AAPCS 5.4 wants composites > 4-bytes returned by arg except for // Homogeneous Aggregates of up-to 4 float types (6.1.2.1) - an HFA. // TODO: see if Tsarray should be candidate for HFA. if (tf->isref) return false; Type *rt = tf->next->toBasetype(); if (!isPOD(rt)) return true; return rt->ty == Tsarray || (rt->ty == Tstruct && rt->size() > 4 && (gTargetMachine->Options.FloatABIType == llvm::FloatABI::Soft || !isHFA((TypeStruct *)rt))); } bool passByVal(TypeFunction *, Type *t) override { // AAPCS does not use an indirect arg to pass aggregates, however // clang uses byval for types > 64-bytes, then llvm backend // converts back to non-byval. Without this special handling the // optimzer generates bad code (e.g. std.random unittest crash). t = t->toBasetype(); return ((t->ty == Tsarray || t->ty == Tstruct) && t->size() > 64); // Note: byval can have a codegen problem with -O1 and higher. // What happens is that load instructions are being incorrectly // reordered before stores. It is a problem in the LLVM backend. // The outcome is a program with incorrect results or crashes. // It happens in the "top-down list latency scheduler" pass // // https://forum.dlang.org/post/m2r3u5ac0c.fsf@comcast.net // // Revist and determine if the byval problem is only for small // structs, say 16-bytes or less, that can entirely fit in // registers. // Note: the codegen is horrible for Tsarrays passed this way - // does a copy without a loop for huge arrays. Could be better if // byval was always used for sarrays, and maybe can if above // problem is better understood. } void rewriteFunctionType(IrFuncTy &fty) override { Type *retTy = fty.ret->type->toBasetype(); if (!fty.ret->byref && retTy->ty == Tstruct) { // Rewrite HFAs only because union HFAs are turned into IR types that are // non-HFA and messes up register selection if (isHFA((TypeStruct *)retTy, &fty.ret->ltype)) { hfaToArray.applyTo(*fty.ret, fty.ret->ltype); } else { integerRewrite.applyTo(*fty.ret); } } for (auto arg : fty.args) { if (!arg->byref) rewriteArgument(fty, *arg); } } void rewriteArgument(IrFuncTy &fty, IrFuncTyArg &arg) override { // structs and arrays need rewrite as i32 arrays. This keeps data layout // unchanged when passed in registers r0-r3 and is necessary to match C ABI // for struct passing. Without out this rewrite, each field or array // element is passed in own register. For example: char[4] now all fits in // r0, where before it consumed r0-r3. Type *ty = arg.type->toBasetype(); // TODO: want to also rewrite Tsarray as i32 arrays, but sometimes // llvm selects an aligned ldrd instruction even though the ptr is // unaligned (e.g. walking through members of array char[5][]). // if (ty->ty == Tstruct || ty->ty == Tsarray) if (ty->ty == Tstruct) { // Rewrite HFAs only because union HFAs are turned into IR types that are // non-HFA and messes up register selection if (isHFA((TypeStruct *)ty, &arg.ltype)) { hfaToArray.applyTo(arg, arg.ltype); } else if (DtoAlignment(ty) <= 4) { compositeToArray32.applyTo(arg); } else { compositeToArray64.applyTo(arg); } } } Type *vaListType() override { // We need to pass the actual va_list type for correct mangling. Simply // using TypeIdentifier here is a bit wonky but works, as long as the name // is actually available in the scope (this is what DMD does, so if a better // solution is found there, this should be adapted). return createTypeIdentifier(Loc(), Identifier::idPool("__va_list")); } const char *objcMsgSendFunc(Type *ret, IrFuncTy &fty) override { // see objc/message.h for objc_msgSend selection rules if (fty.arg_sret) { return "objc_msgSend_stret"; } return "objc_msgSend"; } }; TargetABI *getArmTargetABI() { return new ArmTargetABI; }