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, 67*0b57cec5SDimitry Andric SmallVectorImpl<std::pair<Use *, Function *>> &Uses, 68*0b57cec5SDimitry Andric SmallPtrSetImpl<Constant *> &ConstantBCs) { 69*0b57cec5SDimitry Andric for (Use &U : V->uses()) { 70*0b57cec5SDimitry Andric if (auto *BC = dyn_cast<BitCastOperator>(U.getUser())) 71*0b57cec5SDimitry Andric findUses(BC, F, Uses, ConstantBCs); 728bcb0991SDimitry Andric else if (auto *A = dyn_cast<GlobalAlias>(U.getUser())) 738bcb0991SDimitry Andric findUses(A, F, Uses, ConstantBCs); 74*0b57cec5SDimitry Andric else if (U.get()->getType() != F.getType()) { 755ffd83dbSDimitry Andric CallBase *CB = dyn_cast<CallBase>(U.getUser()); 765ffd83dbSDimitry Andric if (!CB) 77*0b57cec5SDimitry Andric // Skip uses that aren't immediately called 78*0b57cec5SDimitry Andric continue; 795ffd83dbSDimitry Andric Value *Callee = CB->getCalledOperand(); 80*0b57cec5SDimitry Andric if (Callee != V) 81*0b57cec5SDimitry Andric // Skip calls where the function isn't the callee 82*0b57cec5SDimitry Andric continue; 83*0b57cec5SDimitry Andric if (isa<Constant>(U.get())) { 84*0b57cec5SDimitry Andric // Only add constant bitcasts to the list once; they get RAUW'd 85*0b57cec5SDimitry Andric auto C = ConstantBCs.insert(cast<Constant>(U.get())); 86*0b57cec5SDimitry Andric if (!C.second) 87*0b57cec5SDimitry Andric continue; 88*0b57cec5SDimitry Andric } 89*0b57cec5SDimitry Andric Uses.push_back(std::make_pair(&U, &F)); 90*0b57cec5SDimitry Andric } 91*0b57cec5SDimitry Andric } 92*0b57cec5SDimitry Andric } 93*0b57cec5SDimitry Andric 94*0b57cec5SDimitry Andric // Create a wrapper function with type Ty that calls F (which may have a 95*0b57cec5SDimitry Andric // different type). Attempt to support common bitcasted function idioms: 96*0b57cec5SDimitry Andric // - Call with more arguments than needed: arguments are dropped 97*0b57cec5SDimitry Andric // - Call with fewer arguments than needed: arguments are filled in with undef 98*0b57cec5SDimitry Andric // - Return value is not needed: drop it 99*0b57cec5SDimitry Andric // - Return value needed but not present: supply an undef 100*0b57cec5SDimitry Andric // 101*0b57cec5SDimitry Andric // If the all the argument types of trivially castable to one another (i.e. 102*0b57cec5SDimitry Andric // I32 vs pointer type) then we don't create a wrapper at all (return nullptr 103*0b57cec5SDimitry Andric // instead). 104*0b57cec5SDimitry Andric // 105*0b57cec5SDimitry Andric // If there is a type mismatch that we know would result in an invalid wasm 106*0b57cec5SDimitry Andric // module then generate wrapper that contains unreachable (i.e. abort at 107*0b57cec5SDimitry Andric // runtime). Such programs are deep into undefined behaviour territory, 108*0b57cec5SDimitry Andric // but we choose to fail at runtime rather than generate and invalid module 109*0b57cec5SDimitry Andric // or fail at compiler time. The reason we delay the error is that we want 110*0b57cec5SDimitry Andric // to support the CMake which expects to be able to compile and link programs 111*0b57cec5SDimitry Andric // that refer to functions with entirely incorrect signatures (this is how 112*0b57cec5SDimitry Andric // CMake detects the existence of a function in a toolchain). 113*0b57cec5SDimitry Andric // 114*0b57cec5SDimitry Andric // For bitcasts that involve struct types we don't know at this stage if they 115*0b57cec5SDimitry Andric // would be equivalent at the wasm level and so we can't know if we need to 116*0b57cec5SDimitry Andric // generate a wrapper. 117*0b57cec5SDimitry Andric static Function *createWrapper(Function *F, FunctionType *Ty) { 118*0b57cec5SDimitry Andric Module *M = F->getParent(); 119*0b57cec5SDimitry Andric 120*0b57cec5SDimitry Andric Function *Wrapper = Function::Create(Ty, Function::PrivateLinkage, 121*0b57cec5SDimitry Andric F->getName() + "_bitcast", M); 122*0b57cec5SDimitry Andric BasicBlock *BB = BasicBlock::Create(M->getContext(), "body", Wrapper); 123*0b57cec5SDimitry Andric const DataLayout &DL = BB->getModule()->getDataLayout(); 124*0b57cec5SDimitry Andric 125*0b57cec5SDimitry Andric // Determine what arguments to pass. 126*0b57cec5SDimitry Andric SmallVector<Value *, 4> Args; 127*0b57cec5SDimitry Andric Function::arg_iterator AI = Wrapper->arg_begin(); 128*0b57cec5SDimitry Andric Function::arg_iterator AE = Wrapper->arg_end(); 129*0b57cec5SDimitry Andric FunctionType::param_iterator PI = F->getFunctionType()->param_begin(); 130*0b57cec5SDimitry Andric FunctionType::param_iterator PE = F->getFunctionType()->param_end(); 131*0b57cec5SDimitry Andric bool TypeMismatch = false; 132*0b57cec5SDimitry Andric bool WrapperNeeded = false; 133*0b57cec5SDimitry Andric 134*0b57cec5SDimitry Andric Type *ExpectedRtnType = F->getFunctionType()->getReturnType(); 135*0b57cec5SDimitry Andric Type *RtnType = Ty->getReturnType(); 136*0b57cec5SDimitry Andric 137*0b57cec5SDimitry Andric if ((F->getFunctionType()->getNumParams() != Ty->getNumParams()) || 138*0b57cec5SDimitry Andric (F->getFunctionType()->isVarArg() != Ty->isVarArg()) || 139*0b57cec5SDimitry Andric (ExpectedRtnType != RtnType)) 140*0b57cec5SDimitry Andric WrapperNeeded = true; 141*0b57cec5SDimitry Andric 142*0b57cec5SDimitry Andric for (; AI != AE && PI != PE; ++AI, ++PI) { 143*0b57cec5SDimitry Andric Type *ArgType = AI->getType(); 144*0b57cec5SDimitry Andric Type *ParamType = *PI; 145*0b57cec5SDimitry Andric 146*0b57cec5SDimitry Andric if (ArgType == ParamType) { 147*0b57cec5SDimitry Andric Args.push_back(&*AI); 148*0b57cec5SDimitry Andric } else { 149*0b57cec5SDimitry Andric if (CastInst::isBitOrNoopPointerCastable(ArgType, ParamType, DL)) { 150*0b57cec5SDimitry Andric Instruction *PtrCast = 151*0b57cec5SDimitry Andric CastInst::CreateBitOrPointerCast(AI, ParamType, "cast"); 152*0b57cec5SDimitry Andric BB->getInstList().push_back(PtrCast); 153*0b57cec5SDimitry Andric Args.push_back(PtrCast); 154*0b57cec5SDimitry Andric } else if (ArgType->isStructTy() || ParamType->isStructTy()) { 155*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: struct param type in bitcast: " 156*0b57cec5SDimitry Andric << F->getName() << "\n"); 157*0b57cec5SDimitry Andric WrapperNeeded = false; 158*0b57cec5SDimitry Andric } else { 159*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: arg type mismatch calling: " 160*0b57cec5SDimitry Andric << F->getName() << "\n"); 161*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Arg[" << Args.size() << "] Expected: " 162*0b57cec5SDimitry Andric << *ParamType << " Got: " << *ArgType << "\n"); 163*0b57cec5SDimitry Andric TypeMismatch = true; 164*0b57cec5SDimitry Andric break; 165*0b57cec5SDimitry Andric } 166*0b57cec5SDimitry Andric } 167*0b57cec5SDimitry Andric } 168*0b57cec5SDimitry Andric 169*0b57cec5SDimitry Andric if (WrapperNeeded && !TypeMismatch) { 170*0b57cec5SDimitry Andric for (; PI != PE; ++PI) 171*0b57cec5SDimitry Andric Args.push_back(UndefValue::get(*PI)); 172*0b57cec5SDimitry Andric if (F->isVarArg()) 173*0b57cec5SDimitry Andric for (; AI != AE; ++AI) 174*0b57cec5SDimitry Andric Args.push_back(&*AI); 175*0b57cec5SDimitry Andric 176*0b57cec5SDimitry Andric CallInst *Call = CallInst::Create(F, Args, "", BB); 177*0b57cec5SDimitry Andric 178*0b57cec5SDimitry Andric Type *ExpectedRtnType = F->getFunctionType()->getReturnType(); 179*0b57cec5SDimitry Andric Type *RtnType = Ty->getReturnType(); 180*0b57cec5SDimitry Andric // Determine what value to return. 181*0b57cec5SDimitry Andric if (RtnType->isVoidTy()) { 182*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), BB); 183*0b57cec5SDimitry Andric } else if (ExpectedRtnType->isVoidTy()) { 184*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Creating dummy return: " << *RtnType << "\n"); 185*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), UndefValue::get(RtnType), BB); 186*0b57cec5SDimitry Andric } else if (RtnType == ExpectedRtnType) { 187*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), Call, BB); 188*0b57cec5SDimitry Andric } else if (CastInst::isBitOrNoopPointerCastable(ExpectedRtnType, RtnType, 189*0b57cec5SDimitry Andric DL)) { 190*0b57cec5SDimitry Andric Instruction *Cast = 191*0b57cec5SDimitry Andric CastInst::CreateBitOrPointerCast(Call, RtnType, "cast"); 192*0b57cec5SDimitry Andric BB->getInstList().push_back(Cast); 193*0b57cec5SDimitry Andric ReturnInst::Create(M->getContext(), Cast, BB); 194*0b57cec5SDimitry Andric } else if (RtnType->isStructTy() || ExpectedRtnType->isStructTy()) { 195*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: struct return type in bitcast: " 196*0b57cec5SDimitry Andric << F->getName() << "\n"); 197*0b57cec5SDimitry Andric WrapperNeeded = false; 198*0b57cec5SDimitry Andric } else { 199*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: return type mismatch calling: " 200*0b57cec5SDimitry Andric << F->getName() << "\n"); 201*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Expected: " << *ExpectedRtnType 202*0b57cec5SDimitry Andric << " Got: " << *RtnType << "\n"); 203*0b57cec5SDimitry Andric TypeMismatch = true; 204*0b57cec5SDimitry Andric } 205*0b57cec5SDimitry Andric } 206*0b57cec5SDimitry Andric 207*0b57cec5SDimitry Andric if (TypeMismatch) { 208*0b57cec5SDimitry Andric // Create a new wrapper that simply contains `unreachable`. 209*0b57cec5SDimitry Andric Wrapper->eraseFromParent(); 210*0b57cec5SDimitry Andric Wrapper = Function::Create(Ty, Function::PrivateLinkage, 211*0b57cec5SDimitry Andric F->getName() + "_bitcast_invalid", M); 212*0b57cec5SDimitry Andric BasicBlock *BB = BasicBlock::Create(M->getContext(), "body", Wrapper); 213*0b57cec5SDimitry Andric new UnreachableInst(M->getContext(), BB); 214*0b57cec5SDimitry Andric Wrapper->setName(F->getName() + "_bitcast_invalid"); 215*0b57cec5SDimitry Andric } else if (!WrapperNeeded) { 216*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: no wrapper needed: " << F->getName() 217*0b57cec5SDimitry Andric << "\n"); 218*0b57cec5SDimitry Andric Wrapper->eraseFromParent(); 219*0b57cec5SDimitry Andric return nullptr; 220*0b57cec5SDimitry Andric } 221*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "createWrapper: " << F->getName() << "\n"); 222*0b57cec5SDimitry Andric return Wrapper; 223*0b57cec5SDimitry Andric } 224*0b57cec5SDimitry Andric 225*0b57cec5SDimitry Andric // Test whether a main function with type FuncTy should be rewritten to have 226*0b57cec5SDimitry Andric // type MainTy. 227*0b57cec5SDimitry Andric static bool shouldFixMainFunction(FunctionType *FuncTy, FunctionType *MainTy) { 228*0b57cec5SDimitry Andric // Only fix the main function if it's the standard zero-arg form. That way, 229*0b57cec5SDimitry Andric // the standard cases will work as expected, and users will see signature 230*0b57cec5SDimitry Andric // mismatches from the linker for non-standard cases. 231*0b57cec5SDimitry Andric return FuncTy->getReturnType() == MainTy->getReturnType() && 232*0b57cec5SDimitry Andric FuncTy->getNumParams() == 0 && 233*0b57cec5SDimitry Andric !FuncTy->isVarArg(); 234*0b57cec5SDimitry Andric } 235*0b57cec5SDimitry Andric 236*0b57cec5SDimitry Andric bool FixFunctionBitcasts::runOnModule(Module &M) { 237*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "********** Fix Function Bitcasts **********\n"); 238*0b57cec5SDimitry Andric 239*0b57cec5SDimitry Andric Function *Main = nullptr; 240*0b57cec5SDimitry Andric CallInst *CallMain = nullptr; 241*0b57cec5SDimitry Andric SmallVector<std::pair<Use *, Function *>, 0> Uses; 242*0b57cec5SDimitry Andric SmallPtrSet<Constant *, 2> ConstantBCs; 243*0b57cec5SDimitry Andric 244*0b57cec5SDimitry Andric // Collect all the places that need wrappers. 245*0b57cec5SDimitry Andric for (Function &F : M) { 2465ffd83dbSDimitry Andric // Skip to fix when the function is swiftcc because swiftcc allows 2475ffd83dbSDimitry Andric // bitcast type difference for swiftself and swifterror. 2485ffd83dbSDimitry Andric if (F.getCallingConv() == CallingConv::Swift) 2495ffd83dbSDimitry Andric continue; 250*0b57cec5SDimitry Andric findUses(&F, F, Uses, ConstantBCs); 251*0b57cec5SDimitry Andric 252*0b57cec5SDimitry Andric // If we have a "main" function, and its type isn't 253*0b57cec5SDimitry Andric // "int main(int argc, char *argv[])", create an artificial call with it 254*0b57cec5SDimitry Andric // bitcasted to that type so that we generate a wrapper for it, so that 255*0b57cec5SDimitry Andric // the C runtime can call it. 256*0b57cec5SDimitry Andric if (F.getName() == "main") { 257*0b57cec5SDimitry Andric Main = &F; 258*0b57cec5SDimitry Andric LLVMContext &C = M.getContext(); 259*0b57cec5SDimitry Andric Type *MainArgTys[] = {Type::getInt32Ty(C), 260*0b57cec5SDimitry Andric PointerType::get(Type::getInt8PtrTy(C), 0)}; 261*0b57cec5SDimitry Andric FunctionType *MainTy = FunctionType::get(Type::getInt32Ty(C), MainArgTys, 262*0b57cec5SDimitry Andric /*isVarArg=*/false); 263*0b57cec5SDimitry Andric if (shouldFixMainFunction(F.getFunctionType(), MainTy)) { 264*0b57cec5SDimitry Andric LLVM_DEBUG(dbgs() << "Found `main` function with incorrect type: " 265*0b57cec5SDimitry Andric << *F.getFunctionType() << "\n"); 266*0b57cec5SDimitry Andric Value *Args[] = {UndefValue::get(MainArgTys[0]), 267*0b57cec5SDimitry Andric UndefValue::get(MainArgTys[1])}; 268*0b57cec5SDimitry Andric Value *Casted = 269*0b57cec5SDimitry Andric ConstantExpr::getBitCast(Main, PointerType::get(MainTy, 0)); 270*0b57cec5SDimitry Andric CallMain = CallInst::Create(MainTy, Casted, Args, "call_main"); 271*0b57cec5SDimitry Andric Use *UseMain = &CallMain->getOperandUse(2); 272*0b57cec5SDimitry Andric Uses.push_back(std::make_pair(UseMain, &F)); 273*0b57cec5SDimitry Andric } 274*0b57cec5SDimitry Andric } 275*0b57cec5SDimitry Andric } 276*0b57cec5SDimitry Andric 277*0b57cec5SDimitry Andric DenseMap<std::pair<Function *, FunctionType *>, Function *> Wrappers; 278*0b57cec5SDimitry Andric 279*0b57cec5SDimitry Andric for (auto &UseFunc : Uses) { 280*0b57cec5SDimitry Andric Use *U = UseFunc.first; 281*0b57cec5SDimitry Andric Function *F = UseFunc.second; 282*0b57cec5SDimitry Andric auto *PTy = cast<PointerType>(U->get()->getType()); 283*0b57cec5SDimitry Andric auto *Ty = dyn_cast<FunctionType>(PTy->getElementType()); 284*0b57cec5SDimitry Andric 285*0b57cec5SDimitry Andric // If the function is casted to something like i8* as a "generic pointer" 286*0b57cec5SDimitry Andric // to be later casted to something else, we can't generate a wrapper for it. 287*0b57cec5SDimitry Andric // Just ignore such casts for now. 288*0b57cec5SDimitry Andric if (!Ty) 289*0b57cec5SDimitry Andric continue; 290*0b57cec5SDimitry Andric 291*0b57cec5SDimitry Andric auto Pair = Wrappers.insert(std::make_pair(std::make_pair(F, Ty), nullptr)); 292*0b57cec5SDimitry Andric if (Pair.second) 293*0b57cec5SDimitry Andric Pair.first->second = createWrapper(F, Ty); 294*0b57cec5SDimitry Andric 295*0b57cec5SDimitry Andric Function *Wrapper = Pair.first->second; 296*0b57cec5SDimitry Andric if (!Wrapper) 297*0b57cec5SDimitry Andric continue; 298*0b57cec5SDimitry Andric 299*0b57cec5SDimitry Andric if (isa<Constant>(U->get())) 300*0b57cec5SDimitry Andric U->get()->replaceAllUsesWith(Wrapper); 301*0b57cec5SDimitry Andric else 302*0b57cec5SDimitry Andric U->set(Wrapper); 303*0b57cec5SDimitry Andric } 304*0b57cec5SDimitry Andric 305*0b57cec5SDimitry Andric // If we created a wrapper for main, rename the wrapper so that it's the 306*0b57cec5SDimitry Andric // one that gets called from startup. 307*0b57cec5SDimitry Andric if (CallMain) { 308*0b57cec5SDimitry Andric Main->setName("__original_main"); 309*0b57cec5SDimitry Andric auto *MainWrapper = 3105ffd83dbSDimitry Andric cast<Function>(CallMain->getCalledOperand()->stripPointerCasts()); 311*0b57cec5SDimitry Andric delete CallMain; 312*0b57cec5SDimitry Andric if (Main->isDeclaration()) { 313*0b57cec5SDimitry Andric // The wrapper is not needed in this case as we don't need to export 314*0b57cec5SDimitry Andric // it to anyone else. 315*0b57cec5SDimitry Andric MainWrapper->eraseFromParent(); 316*0b57cec5SDimitry Andric } else { 317*0b57cec5SDimitry Andric // Otherwise give the wrapper the same linkage as the original main 318*0b57cec5SDimitry Andric // function, so that it can be called from the same places. 319*0b57cec5SDimitry Andric MainWrapper->setName("main"); 320*0b57cec5SDimitry Andric MainWrapper->setLinkage(Main->getLinkage()); 321*0b57cec5SDimitry Andric MainWrapper->setVisibility(Main->getVisibility()); 322*0b57cec5SDimitry Andric } 323*0b57cec5SDimitry Andric } 324*0b57cec5SDimitry Andric 325*0b57cec5SDimitry Andric return true; 326*0b57cec5SDimitry Andric } 327