static void addExplicitArguments(std::vector<LLValue *> &args, AttrSet &attrs, IrFuncTy &irFty, LLFunctionType *callableTy, const std::vector<DValue *> &argvals, int numFormalParams) { // Number of arguments added to the LLVM type that are implicit on the // frontend side of things (this, context pointers, etc.) const size_t implicitLLArgCount = args.size(); // Number of formal arguments in the LLVM type (i.e. excluding varargs). const size_t formalLLArgCount = irFty.args.size(); // The number of explicit arguments in the D call expression (including // varargs), not all of which necessarily generate a LLVM argument. const size_t explicitDArgCount = argvals.size(); // construct and initialize an IrFuncTyArg object for each vararg std::vector<IrFuncTyArg *> optionalIrArgs; for (size_t i = numFormalParams; i < explicitDArgCount; i++) { Type *argType = argvals[i]->getType(); bool passByVal = gABI->passByVal(argType); AttrBuilder initialAttrs; if (passByVal) { initialAttrs.add(LLAttribute::ByVal); } else { initialAttrs.add(DtoShouldExtend(argType)); } optionalIrArgs.push_back(new IrFuncTyArg(argType, passByVal, initialAttrs)); optionalIrArgs.back()->parametersIdx = i; } // let the ABI rewrite the IrFuncTyArg objects gABI->rewriteVarargs(irFty, optionalIrArgs); const size_t explicitLLArgCount = formalLLArgCount + optionalIrArgs.size(); args.resize(implicitLLArgCount + explicitLLArgCount, static_cast<llvm::Value *>(nullptr)); // Iterate the explicit arguments from left to right in the D source, // which is the reverse of the LLVM order if irFty.reverseParams is true. for (size_t i = 0; i < explicitLLArgCount; ++i) { const bool isVararg = (i >= irFty.args.size()); IrFuncTyArg *irArg = nullptr; if (isVararg) { irArg = optionalIrArgs[i - numFormalParams]; } else { irArg = irFty.args[i]; } DValue *const argval = argvals[irArg->parametersIdx]; Type *const argType = argval->getType(); llvm::Value *llVal = nullptr; if (isVararg) { llVal = irFty.putParam(*irArg, argval); } else { llVal = irFty.putParam(i, argval); } const size_t llArgIdx = implicitLLArgCount + (irFty.reverseParams ? explicitLLArgCount - i - 1 : i); llvm::Type *const callableArgType = (isVararg ? nullptr : callableTy->getParamType(llArgIdx)); // Hack around LDC assuming structs and static arrays are in memory: // If the function wants a struct, and the argument value is a // pointer to a struct, load from it before passing it in. if (isaPointer(llVal) && DtoIsPassedByRef(argType) && ((!isVararg && !isaPointer(callableArgType)) || (isVararg && !irArg->byref && !irArg->isByVal()))) { Logger::println("Loading struct type for function argument"); llVal = DtoLoad(llVal); } // parameter type mismatch, this is hard to get rid of if (!isVararg && llVal->getType() != callableArgType) { IF_LOG { Logger::cout() << "arg: " << *llVal << '\n'; Logger::cout() << "expects: " << *callableArgType << '\n'; } if (isaStruct(llVal)) { llVal = DtoAggrPaint(llVal, callableArgType); } else { llVal = DtoBitCast(llVal, callableArgType); } } args[llArgIdx] = llVal; // +1 as index 0 contains the function attributes. attrs.add(llArgIdx + 1, irArg->attrs); if (isVararg) { delete irArg; } }
llvm::FunctionType* DtoFunctionType(Type* type, IrFuncTy &irFty, Type* thistype, Type* nesttype, bool isMain, bool isCtor, bool isIntrinsic) { IF_LOG Logger::println("DtoFunctionType(%s)", type->toChars()); LOG_SCOPE // sanity check assert(type->ty == Tfunction); TypeFunction* f = static_cast<TypeFunction*>(type); assert(f->next && "Encountered function type with invalid return type; " "trying to codegen function ignored by the frontend?"); // Return cached type if available if (irFty.funcType) return irFty.funcType; TargetABI* abi = (isIntrinsic ? TargetABI::getIntrinsic() : gABI); // Do not modify irFty yet; this function may be called recursively if any // of the argument types refer to this type. IrFuncTy newIrFty; // llvm idx counter size_t lidx = 0; // main needs a little special handling if (isMain) { newIrFty.ret = new IrFuncTyArg(Type::tint32, false); } // sane return value else { Type* rt = f->next; AttrBuilder attrBuilder; // sret return if (abi->returnInArg(f)) { newIrFty.arg_sret = new IrFuncTyArg(rt, true, AttrBuilder().add(LDC_ATTRIBUTE(StructRet)).add(LDC_ATTRIBUTE(NoAlias))); rt = Type::tvoid; lidx++; } // sext/zext return else { Type *t = rt; if (f->isref) t = t->pointerTo(); attrBuilder.add(DtoShouldExtend(t)); } newIrFty.ret = new IrFuncTyArg(rt, f->isref, attrBuilder); } lidx++; // member functions if (thistype) { AttrBuilder attrBuilder; #if LDC_LLVM_VER >= 303 if (isCtor) attrBuilder.add(LDC_ATTRIBUTE(Returned)); #endif newIrFty.arg_this = new IrFuncTyArg(thistype, thistype->toBasetype()->ty == Tstruct, attrBuilder); lidx++; } // and nested functions else if (nesttype) { newIrFty.arg_nest = new IrFuncTyArg(nesttype, false); lidx++; } // vararg functions are special too if (f->varargs) { if (f->linkage == LINKd) { // d style with hidden args // 2 (array) is handled by the frontend if (f->varargs == 1) { // _arguments newIrFty.arg_arguments = new IrFuncTyArg(Type::dtypeinfo->type->arrayOf(), false); lidx++; } } newIrFty.c_vararg = true; } // if this _Dmain() doesn't have an argument, we force it to have one int nargs = Parameter::dim(f->parameters); if (isMain && nargs == 0) { Type* mainargs = Type::tchar->arrayOf()->arrayOf(); newIrFty.args.push_back(new IrFuncTyArg(mainargs, false)); lidx++; } // add explicit parameters else for (int i = 0; i < nargs; i++) { // get argument Parameter* arg = Parameter::getNth(f->parameters, i); // reference semantics? ref, out and d1 static arrays are bool byref = arg->storageClass & (STCref|STCout); Type* argtype = arg->type; AttrBuilder attrBuilder; // handle lazy args if (arg->storageClass & STClazy) { Logger::println("lazy param"); TypeFunction *ltf = new TypeFunction(NULL, arg->type, 0, LINKd); TypeDelegate *ltd = new TypeDelegate(ltf); argtype = ltd; } else if (!byref) { // byval if (abi->passByVal(argtype)) { attrBuilder.add(LDC_ATTRIBUTE(ByVal)); // set byref, because byval requires a pointed LLVM type byref = true; } // sext/zext else { attrBuilder.add(DtoShouldExtend(argtype)); } } newIrFty.args.push_back(new IrFuncTyArg(argtype, byref, attrBuilder)); lidx++; } // let the abi rewrite the types as necesary abi->rewriteFunctionType(f, newIrFty); // Now we can modify irFty safely. irFty = llvm_move(newIrFty); // build the function type std::vector<LLType*> argtypes; argtypes.reserve(lidx); if (irFty.arg_sret) argtypes.push_back(irFty.arg_sret->ltype); if (irFty.arg_this) argtypes.push_back(irFty.arg_this->ltype); if (irFty.arg_nest) argtypes.push_back(irFty.arg_nest->ltype); if (irFty.arg_arguments) argtypes.push_back(irFty.arg_arguments->ltype); size_t beg = argtypes.size(); size_t nargs2 = irFty.args.size(); for (size_t i = 0; i < nargs2; i++) { argtypes.push_back(irFty.args[i]->ltype); } // reverse params? if (irFty.reverseParams && nargs2 > 1) { std::reverse(argtypes.begin() + beg, argtypes.end()); } irFty.funcType = LLFunctionType::get(irFty.ret->ltype, argtypes, irFty.c_vararg); IF_LOG Logger::cout() << "Final function type: " << *irFty.funcType << "\n"; return irFty.funcType; }