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