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