1*0b57cec5SDimitry Andric //===-- WebAssemblyFixFunctionBitcasts.cpp - Fix function bitcasts --------===// 2*0b57cec5SDimitry Andric // 3*0b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*0b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5*0b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*0b57cec5SDimitry Andric // 7*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 8*0b57cec5SDimitry Andric /// 9*0b57cec5SDimitry Andric /// \file 10*0b57cec5SDimitry Andric /// Fix bitcasted functions. 11*0b57cec5SDimitry Andric /// 12*0b57cec5SDimitry Andric /// WebAssembly requires caller and callee signatures to match, however in LLVM, 13*0b57cec5SDimitry Andric /// some amount of slop is vaguely permitted. Detect mismatch by looking for 14*0b57cec5SDimitry Andric /// bitcasts of functions and rewrite them to use wrapper functions instead. 15*0b57cec5SDimitry Andric /// 16*0b57cec5SDimitry Andric /// This doesn't catch all cases, such as when a function's address is taken in 17*0b57cec5SDimitry Andric /// one place and casted in another, but it works for many common cases. 18*0b57cec5SDimitry Andric /// 19*0b57cec5SDimitry Andric /// Note that LLVM already optimizes away function bitcasts in common cases by 20*0b57cec5SDimitry Andric /// dropping arguments as needed, so this pass only ends up getting used in less 21*0b57cec5SDimitry Andric /// common cases. 22*0b57cec5SDimitry Andric /// 23*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 24*0b57cec5SDimitry Andric 25*0b57cec5SDimitry Andric #include "WebAssembly.h" 26*0b57cec5SDimitry Andric #include "llvm/IR/Constants.h" 27*0b57cec5SDimitry Andric #include "llvm/IR/Instructions.h" 28*0b57cec5SDimitry Andric #include "llvm/IR/Module.h" 29*0b57cec5SDimitry Andric #include "llvm/IR/Operator.h" 30*0b57cec5SDimitry Andric #include "llvm/Pass.h" 31*0b57cec5SDimitry Andric #include "llvm/Support/Debug.h" 32*0b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h" 33*0b57cec5SDimitry Andric using namespace llvm; 34*0b57cec5SDimitry Andric 35*0b57cec5SDimitry Andric #define DEBUG_TYPE "wasm-fix-function-bitcasts" 36*0b57cec5SDimitry Andric 37*0b57cec5SDimitry Andric namespace { 38*0b57cec5SDimitry Andric class FixFunctionBitcasts final : public ModulePass { 39*0b57cec5SDimitry Andric StringRef getPassName() const override { 40*0b57cec5SDimitry Andric return "WebAssembly Fix Function Bitcasts"; 41*0b57cec5SDimitry Andric } 42*0b57cec5SDimitry Andric 43*0b57cec5SDimitry Andric void getAnalysisUsage(AnalysisUsage &AU) const override { 44*0b57cec5SDimitry Andric AU.setPreservesCFG(); 45*0b57cec5SDimitry Andric ModulePass::getAnalysisUsage(AU); 46*0b57cec5SDimitry Andric } 47*0b57cec5SDimitry Andric 48*0b57cec5SDimitry Andric bool runOnModule(Module &M) override; 49*0b57cec5SDimitry Andric 50*0b57cec5SDimitry Andric public: 51*0b57cec5SDimitry Andric static char ID; 52*0b57cec5SDimitry Andric FixFunctionBitcasts() : ModulePass(ID) {} 53*0b57cec5SDimitry Andric }; 54*0b57cec5SDimitry Andric } // End anonymous namespace 55*0b57cec5SDimitry Andric 56*0b57cec5SDimitry Andric char FixFunctionBitcasts::ID = 0; 57*0b57cec5SDimitry Andric INITIALIZE_PASS(FixFunctionBitcasts, DEBUG_TYPE, 58*0b57cec5SDimitry Andric "Fix mismatching bitcasts for WebAssembly", false, false) 59*0b57cec5SDimitry Andric 60*0b57cec5SDimitry Andric ModulePass *llvm::createWebAssemblyFixFunctionBitcasts() { 61*0b57cec5SDimitry Andric return new FixFunctionBitcasts(); 62*0b57cec5SDimitry Andric } 63*0b57cec5SDimitry Andric 64*0b57cec5SDimitry Andric // Recursively descend the def-use lists from V to find non-bitcast users of 65*0b57cec5SDimitry Andric // bitcasts of V. 66*0b57cec5SDimitry Andric static void findUses(Value *V, Function &F, 67349cc55cSDimitry Andric SmallVectorImpl<std::pair<CallBase *, Function *>> &Uses) { 68349cc55cSDimitry Andric for (User *U : V->users()) { 69349cc55cSDimitry Andric if (auto *BC = dyn_cast<BitCastOperator>(U)) 70349cc55cSDimitry Andric findUses(BC, F, Uses); 71349cc55cSDimitry Andric else if (auto *A = dyn_cast<GlobalAlias>(U)) 72349cc55cSDimitry Andric findUses(A, F, Uses); 73349cc55cSDimitry Andric else if (auto *CB = dyn_cast<CallBase>(U)) { 745ffd83dbSDimitry Andric Value *Callee = CB->getCalledOperand(); 75*0b57cec5SDimitry Andric if (Callee != V) 76*0b57cec5SDimitry Andric // Skip calls where the function isn't the callee 77*0b57cec5SDimitry Andric continue; 78349cc55cSDimitry Andric if (CB->getFunctionType() == F.getValueType()) 79349cc55cSDimitry Andric // Skip uses that are immediately called 80*0b57cec5SDimitry Andric continue; 81349cc55cSDimitry Andric Uses.push_back(std::make_pair(CB, &F)); 82*0b57cec5SDimitry Andric } 83*0b57cec5SDimitry Andric } 84*0b57cec5SDimitry Andric } 85*0b57cec5SDimitry Andric 86*0b57cec5SDimitry Andric // Create a wrapper function with type Ty that calls F (which may have a 87*0b57cec5SDimitry Andric // different type). Attempt to support common bitcasted function idioms: 88*0b57cec5SDimitry Andric // - Call with more arguments than needed: arguments are dropped 89*0b57cec5SDimitry Andric // - Call with fewer arguments than needed: arguments are filled in with undef 90*0b57cec5SDimitry Andric // - Return value is not needed: drop it 91*0b57cec5SDimitry Andric // - Return value needed but not present: supply an undef 92*0b57cec5SDimitry Andric // 93*0b57cec5SDimitry Andric // If the all the argument types of trivially castable to one another (i.e. 94*0b57cec5SDimitry Andric // I32 vs pointer type) then we don't create a wrapper at all (return nullptr 95*0b57cec5SDimitry Andric // instead). 96*0b57cec5SDimitry Andric // 97*0b57cec5SDimitry Andric // If there is a type mismatch that we know would result in an invalid wasm 98*0b57cec5SDimitry Andric // module then generate wrapper that contains unreachable (i.e. abort at 99*0b57cec5SDimitry Andric // runtime). Such programs are deep into undefined behaviour territory, 100*0b57cec5SDimitry Andric // but we choose to fail at runtime rather than generate and invalid module 101*0b57cec5SDimitry Andric // or fail at compiler time. The reason we delay the error is that we want 102*0b57cec5SDimitry Andric // to support the CMake which expects to be able to compile and link programs 103*0b57cec5SDimitry Andric // that refer to functions with entirely incorrect signatures (this is how 104*0b57cec5SDimitry Andric // CMake detects the existence of a function in a toolchain). 105*0b57cec5SDimitry Andric // 106*0b57cec5SDimitry Andric // For bitcasts that involve struct types we don't know at this stage if they 107*0b57cec5SDimitry Andric // would be equivalent at the wasm level and so we can't know if we need to 108*0b57cec5SDimitry Andric // generate a wrapper. 109*0b57cec5SDimitry Andric static Function *createWrapper(Function *F, FunctionType *Ty) { 110*0b57cec5SDimitry Andric Module *M = F->getParent(); 111*0b57cec5SDimitry Andric 112*0b57cec5SDimitry Andric Function *Wrapper = Function::Create(Ty, Function::PrivateLinkage, 113*0b57cec5SDimitry Andric F->getName() + "_bitcast", M); 114*0b57cec5SDimitry Andric BasicBlock *BB = BasicBlock::Create(M->getContext(), "body", Wrapper); 115*0b57cec5SDimitry Andric const DataLayout &DL = BB->getModule()->getDataLayout(); 116*0b57cec5SDimitry Andric 117*0b57cec5SDimitry Andric // Determine what arguments to pass. 118*0b57cec5SDimitry Andric SmallVector<Value *, 4> Args; 119*0b57cec5SDimitry Andric Function::arg_iterator AI = Wrapper->arg_begin(); 120*0b57cec5SDimitry Andric Function::arg_iterator AE = Wrapper->arg_end(); 121*0b57cec5SDimitry Andric FunctionType::param_iterator PI = F->getFunctionType()->param_begin(); 122*0b57cec5SDimitry Andric FunctionType::param_iterator PE = F->getFunctionType()->param_end(); 123*0b57cec5SDimitry Andric bool TypeMismatch = false; 124*0b57cec5SDimitry Andric bool WrapperNeeded = false; 125*0b57cec5SDimitry Andric 126*0b57cec5SDimitry Andric Type *ExpectedRtnType = F->getFunctionType()->getReturnType(); 127*0b57cec5SDimitry Andric Type *RtnType = Ty->getReturnType(); 128*0b57cec5SDimitry Andric 129*0b57cec5SDimitry Andric if ((F->getFunctionType()->getNumParams() != Ty->getNumParams()) || 130*0b57cec5SDimitry Andric (F->getFunctionType()->isVarArg() != Ty->isVarArg()) || 131*0b57cec5SDimitry Andric (ExpectedRtnType != RtnType)) 132*0b57cec5SDimitry Andric WrapperNeeded = true; 133*0b57cec5SDimitry Andric 134*0b57cec5SDimitry Andric for (; AI != AE && PI != PE; ++AI, ++PI) { 135*0b57cec5SDimitry Andric Type *ArgType = AI->getType(); 136*0b57cec5SDimitry Andric Type *ParamType = *PI; 137*0b57cec5SDimitry Andric 138*0b57cec5SDimitry Andric if (ArgType == ParamType) { 139*0b57cec5SDimitry Andric Args.push_back(&*AI); 140*0b57cec5SDimitry Andric } else { 141*0b57cec5SDimitry Andric if (CastInst::isBitOrNoopPointerCastable(ArgType, ParamType, DL)) { 142*0b57cec5SDimitry Andric Instruction *PtrCast = 143*0b57cec5SDimitry Andric CastInst::CreateBitOrPointerCast(AI, ParamType, "cast"); 144bdd1243dSDimitry Andric PtrCast->insertInto(BB, BB->end()); 145*0b57cec5SDimitry Andric Args.push_back(PtrCast); 146*0b57cec5SDimitry Andric } else if (ArgType->isStructTy() || ParamType->isStructTy()) { 147*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: struct param type in bitcast: " 148*0b57cec5SDimitry Andric << F->getName() << "\n"); 149*0b57cec5SDimitry Andric WrapperNeeded = false; 150*0b57cec5SDimitry Andric } else { 151*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: arg type mismatch calling: " 152*0b57cec5SDimitry Andric << F->getName() << "\n"); 153*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Arg[" << Args.size() << "] Expected: " 154*0b57cec5SDimitry Andric << *ParamType << " Got: " << *ArgType << "\n"); 155*0b57cec5SDimitry Andric TypeMismatch = true; 156*0b57cec5SDimitry Andric break; 157*0b57cec5SDimitry Andric } 158*0b57cec5SDimitry Andric } 159*0b57cec5SDimitry Andric } 160*0b57cec5SDimitry Andric 161*0b57cec5SDimitry Andric if (WrapperNeeded && !TypeMismatch) { 162*0b57cec5SDimitry Andric for (; PI != PE; ++PI) 163*0b57cec5SDimitry Andric Args.push_back(UndefValue::get(*PI)); 164*0b57cec5SDimitry Andric if (F->isVarArg()) 165*0b57cec5SDimitry Andric for (; AI != AE; ++AI) 166*0b57cec5SDimitry Andric Args.push_back(&*AI); 167*0b57cec5SDimitry Andric 168*0b57cec5SDimitry Andric CallInst *Call = CallInst::Create(F, Args, "", BB); 169*0b57cec5SDimitry Andric 170*0b57cec5SDimitry Andric Type *ExpectedRtnType = F->getFunctionType()->getReturnType(); 171*0b57cec5SDimitry Andric Type *RtnType = Ty->getReturnType(); 172*0b57cec5SDimitry Andric // Determine what value to return. 173*0b57cec5SDimitry Andric if (RtnType->isVoidTy()) { 174*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), BB); 175*0b57cec5SDimitry Andric } else if (ExpectedRtnType->isVoidTy()) { 176*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Creating dummy return: " << *RtnType << "\n"); 177*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), UndefValue::get(RtnType), BB); 178*0b57cec5SDimitry Andric } else if (RtnType == ExpectedRtnType) { 179*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), Call, BB); 180*0b57cec5SDimitry Andric } else if (CastInst::isBitOrNoopPointerCastable(ExpectedRtnType, RtnType, 181*0b57cec5SDimitry Andric DL)) { 182*0b57cec5SDimitry Andric Instruction *Cast = 183*0b57cec5SDimitry Andric CastInst::CreateBitOrPointerCast(Call, RtnType, "cast"); 184bdd1243dSDimitry Andric Cast->insertInto(BB, BB->end()); 185*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), Cast, BB); 186*0b57cec5SDimitry Andric } else if (RtnType->isStructTy() || ExpectedRtnType->isStructTy()) { 187*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: struct return type in bitcast: " 188*0b57cec5SDimitry Andric << F->getName() << "\n"); 189*0b57cec5SDimitry Andric WrapperNeeded = false; 190*0b57cec5SDimitry Andric } else { 191*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: return type mismatch calling: " 192*0b57cec5SDimitry Andric << F->getName() << "\n"); 193*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Expected: " << *ExpectedRtnType 194*0b57cec5SDimitry Andric << " Got: " << *RtnType << "\n"); 195*0b57cec5SDimitry Andric TypeMismatch = true; 196*0b57cec5SDimitry Andric } 197*0b57cec5SDimitry Andric } 198*0b57cec5SDimitry Andric 199*0b57cec5SDimitry Andric if (TypeMismatch) { 200*0b57cec5SDimitry Andric // Create a new wrapper that simply contains `unreachable`. 201*0b57cec5SDimitry Andric Wrapper->eraseFromParent(); 202*0b57cec5SDimitry Andric Wrapper = Function::Create(Ty, Function::PrivateLinkage, 203*0b57cec5SDimitry Andric F->getName() + "_bitcast_invalid", M); 204*0b57cec5SDimitry Andric BasicBlock *BB = BasicBlock::Create(M->getContext(), "body", Wrapper); 205*0b57cec5SDimitry Andric new UnreachableInst(M->getContext(), BB); 206*0b57cec5SDimitry Andric Wrapper->setName(F->getName() + "_bitcast_invalid"); 207*0b57cec5SDimitry Andric } else if (!WrapperNeeded) { 208*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: no wrapper needed: " << F->getName() 209*0b57cec5SDimitry Andric << "\n"); 210*0b57cec5SDimitry Andric Wrapper->eraseFromParent(); 211*0b57cec5SDimitry Andric return nullptr; 212*0b57cec5SDimitry Andric } 213*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: " << F->getName() << "\n"); 214*0b57cec5SDimitry Andric return Wrapper; 215*0b57cec5SDimitry Andric } 216*0b57cec5SDimitry Andric 217*0b57cec5SDimitry Andric // Test whether a main function with type FuncTy should be rewritten to have 218*0b57cec5SDimitry Andric // type MainTy. 219*0b57cec5SDimitry Andric static bool shouldFixMainFunction(FunctionType *FuncTy, FunctionType *MainTy) { 220*0b57cec5SDimitry Andric // Only fix the main function if it's the standard zero-arg form. That way, 221*0b57cec5SDimitry Andric // the standard cases will work as expected, and users will see signature 222*0b57cec5SDimitry Andric // mismatches from the linker for non-standard cases. 223*0b57cec5SDimitry Andric return FuncTy->getReturnType() == MainTy->getReturnType() && 224*0b57cec5SDimitry Andric FuncTy->getNumParams() == 0 && 225*0b57cec5SDimitry Andric !FuncTy->isVarArg(); 226*0b57cec5SDimitry Andric } 227*0b57cec5SDimitry Andric 228*0b57cec5SDimitry Andric bool FixFunctionBitcasts::runOnModule(Module &M) { 229*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "********** Fix Function Bitcasts **********\n"); 230*0b57cec5SDimitry Andric 231*0b57cec5SDimitry Andric Function *Main = nullptr; 232*0b57cec5SDimitry Andric CallInst *CallMain = nullptr; 233349cc55cSDimitry Andric SmallVector<std::pair<CallBase *, Function *>, 0> Uses; 234*0b57cec5SDimitry Andric 235*0b57cec5SDimitry Andric // Collect all the places that need wrappers. 236*0b57cec5SDimitry Andric for (Function &F : M) { 2375ffd83dbSDimitry Andric // Skip to fix when the function is swiftcc because swiftcc allows 2385ffd83dbSDimitry Andric // bitcast type difference for swiftself and swifterror. 2395ffd83dbSDimitry Andric if (F.getCallingConv() == CallingConv::Swift) 2405ffd83dbSDimitry Andric continue; 241349cc55cSDimitry Andric findUses(&F, F, Uses); 242*0b57cec5SDimitry Andric 243*0b57cec5SDimitry Andric // If we have a "main" function, and its type isn't 244*0b57cec5SDimitry Andric // "int main(int argc, char *argv[])", create an artificial call with it 245*0b57cec5SDimitry Andric // bitcasted to that type so that we generate a wrapper for it, so that 246*0b57cec5SDimitry Andric // the C runtime can call it. 247*0b57cec5SDimitry Andric if (F.getName() == "main") { 248*0b57cec5SDimitry Andric Main = &F; 249*0b57cec5SDimitry Andric LLVMContext &C = M.getContext(); 250*0b57cec5SDimitry Andric Type *MainArgTys[] = {Type::getInt32Ty(C), 251*0b57cec5SDimitry Andric PointerType::get(Type::getInt8PtrTy(C), 0)}; 252*0b57cec5SDimitry Andric FunctionType *MainTy = FunctionType::get(Type::getInt32Ty(C), MainArgTys, 253*0b57cec5SDimitry Andric /*isVarArg=*/false); 254*0b57cec5SDimitry Andric if (shouldFixMainFunction(F.getFunctionType(), MainTy)) { 255*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Found `main` function with incorrect type: " 256*0b57cec5SDimitry Andric << *F.getFunctionType() << "\n"); 257*0b57cec5SDimitry Andric Value *Args[] = {UndefValue::get(MainArgTys[0]), 258*0b57cec5SDimitry Andric UndefValue::get(MainArgTys[1])}; 259*0b57cec5SDimitry Andric Value *Casted = 260*0b57cec5SDimitry Andric ConstantExpr::getBitCast(Main, PointerType::get(MainTy, 0)); 261*0b57cec5SDimitry Andric CallMain = CallInst::Create(MainTy, Casted, Args, "call_main"); 262349cc55cSDimitry Andric Uses.push_back(std::make_pair(CallMain, &F)); 263*0b57cec5SDimitry Andric } 264*0b57cec5SDimitry Andric } 265*0b57cec5SDimitry Andric } 266*0b57cec5SDimitry Andric 267*0b57cec5SDimitry Andric DenseMap<std::pair<Function *, FunctionType *>, Function *> Wrappers; 268*0b57cec5SDimitry Andric 269*0b57cec5SDimitry Andric for (auto &UseFunc : Uses) { 270349cc55cSDimitry Andric CallBase *CB = UseFunc.first; 271*0b57cec5SDimitry Andric Function *F = UseFunc.second; 272349cc55cSDimitry Andric FunctionType *Ty = CB->getFunctionType(); 273*0b57cec5SDimitry Andric 274*0b57cec5SDimitry Andric auto Pair = Wrappers.insert(std::make_pair(std::make_pair(F, Ty), nullptr)); 275*0b57cec5SDimitry Andric if (Pair.second) 276*0b57cec5SDimitry Andric Pair.first->second = createWrapper(F, Ty); 277*0b57cec5SDimitry Andric 278*0b57cec5SDimitry Andric Function *Wrapper = Pair.first->second; 279*0b57cec5SDimitry Andric if (!Wrapper) 280*0b57cec5SDimitry Andric continue; 281*0b57cec5SDimitry Andric 282349cc55cSDimitry Andric CB->setCalledOperand(Wrapper); 283*0b57cec5SDimitry Andric } 284*0b57cec5SDimitry Andric 285*0b57cec5SDimitry Andric // If we created a wrapper for main, rename the wrapper so that it's the 286*0b57cec5SDimitry Andric // one that gets called from startup. 287*0b57cec5SDimitry Andric if (CallMain) { 288*0b57cec5SDimitry Andric Main->setName("__original_main"); 289*0b57cec5SDimitry Andric auto *MainWrapper = 2905ffd83dbSDimitry Andric cast<Function>(CallMain->getCalledOperand()->stripPointerCasts()); 291*0b57cec5SDimitry Andric delete CallMain; 292*0b57cec5SDimitry Andric if (Main->isDeclaration()) { 293*0b57cec5SDimitry Andric // The wrapper is not needed in this case as we don't need to export 294*0b57cec5SDimitry Andric // it to anyone else. 295*0b57cec5SDimitry Andric MainWrapper->eraseFromParent(); 296*0b57cec5SDimitry Andric } else { 297*0b57cec5SDimitry Andric // Otherwise give the wrapper the same linkage as the original main 298*0b57cec5SDimitry Andric // function, so that it can be called from the same places. 299*0b57cec5SDimitry Andric MainWrapper->setName("main"); 300*0b57cec5SDimitry Andric MainWrapper->setLinkage(Main->getLinkage()); 301*0b57cec5SDimitry Andric MainWrapper->setVisibility(Main->getVisibility()); 302*0b57cec5SDimitry Andric } 303*0b57cec5SDimitry Andric } 304*0b57cec5SDimitry Andric 305*0b57cec5SDimitry Andric return true; 306*0b57cec5SDimitry Andric } 307