/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef _WIN32 #include #include #include #else #include #endif #ifdef ANDROID #include #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef _WIN32 #include #endif #include #include #include #include #include #include #include #include #include #if OSL_DEBUG_LEVEL > 0 #include #include #endif #ifdef LINUX #include #endif using namespace ::com::sun::star; static bool g_bIsLeanException; static oslSignalAction VCLExceptionSignal_impl( void* /*pData*/, oslSignalInfo* pInfo) { static volatile bool bIn = false; // if we crash again, bail out immediately if ( bIn || g_bIsLeanException) return osl_Signal_ActCallNextHdl; ExceptionCategory nVCLException = ExceptionCategory::NONE; // UAE if ( (pInfo->Signal == osl_Signal_AccessViolation) || (pInfo->Signal == osl_Signal_IntegerDivideByZero) || (pInfo->Signal == osl_Signal_FloatDivideByZero) || (pInfo->Signal == osl_Signal_DebugBreak) ) { nVCLException = ExceptionCategory::System; #if HAVE_FEATURE_OPENGL if (OpenGLZone::isInZone()) OpenGLZone::hardDisable(); #endif #if HAVE_FEATURE_SKIA if (SkiaZone::isInZone()) SkiaZone::hardDisable(); #endif #if HAVE_FEATURE_OPENCL if (OpenCLZone::isInZone()) { OpenCLZone::hardDisable(); #ifdef _WIN32 if (OpenCLInitialZone::isInZone()) TerminateProcess(GetCurrentProcess(), EXITHELPER_NORMAL_RESTART); #endif } #endif } // DISPLAY-Unix if ((pInfo->Signal == osl_Signal_User) && (pInfo->UserSignal == OSL_SIGNAL_USER_X11SUBSYSTEMERROR) ) nVCLException = ExceptionCategory::UserInterface; if ( nVCLException != ExceptionCategory::NONE ) { bIn = true; vcl::SolarMutexTryAndBuyGuard aLock; if( aLock.isAcquired()) { // do not stop timer because otherwise the UAE-Box will not be painted as well ImplSVData* pSVData = ImplGetSVData(); if ( pSVData->mpApp ) { SystemWindowFlags nOldMode = Application::GetSystemWindowMode(); Application::SetSystemWindowMode( nOldMode & ~SystemWindowFlags::NOAUTOMODE ); pSVData->mpApp->Exception( nVCLException ); Application::SetSystemWindowMode( nOldMode ); } } bIn = false; } return osl_Signal_ActCallNextHdl; } int ImplSVMain() { // The 'real' SVMain() ImplSVData* pSVData = ImplGetSVData(); SAL_WARN_IF( !pSVData->mpApp, "vcl", "no instance of class Application" ); int nReturn = EXIT_FAILURE; const bool bWasInitVCL = IsVCLInit(); const bool bInit = bWasInitVCL || InitVCL(); int nRet = 0; if (!bWasInitVCL && bInit && pSVData->mpDefInst->SVMainHook(&nRet)) return nRet; if( bInit ) { // call application main pSVData->maAppData.mbInAppMain = true; nReturn = pSVData->mpApp->Main(); pSVData->maAppData.mbInAppMain = false; } if( pSVData->mxDisplayConnection.is() ) { pSVData->mxDisplayConnection->terminate(); pSVData->mxDisplayConnection.clear(); } // This is a hack to work around the problem of the asynchronous nature // of bridging accessibility through Java: on shutdown there might still // be some events in the AWT EventQueue, which need the SolarMutex which // - on the other hand - is destroyed in DeInitVCL(). So empty the queue // here .. if( pSVData->mxAccessBridge.is() ) { { SolarMutexReleaser aReleaser; pSVData->mxAccessBridge->dispose(); } pSVData->mxAccessBridge.clear(); } WatchdogThread::stop(); DeInitVCL(); return nReturn; } int SVMain() { return ImplSVMain(); } // This variable is set when no Application object has been instantiated // before InitVCL is called static Application * pOwnSvApp = nullptr; // Exception handler. pExceptionHandler != NULL => VCL already inited static oslSignalHandler pExceptionHandler = nullptr; namespace { class DesktopEnvironmentContext: public cppu::WeakImplHelper< css::uno::XCurrentContext > { public: explicit DesktopEnvironmentContext( const css::uno::Reference< css::uno::XCurrentContext > & ctx) : m_xNextContext( ctx ) {} // XCurrentContext virtual css::uno::Any SAL_CALL getValueByName( const OUString& Name ) override; private: css::uno::Reference< css::uno::XCurrentContext > m_xNextContext; }; } uno::Any SAL_CALL DesktopEnvironmentContext::getValueByName( const OUString& Name) { uno::Any retVal; if ( Name == "system.desktop-environment" ) { retVal <<= Application::GetDesktopEnvironment(); } else if( m_xNextContext.is() ) { // Call next context in chain if found retVal = m_xNextContext->getValueByName( Name ); } return retVal; } bool IsVCLInit() { ImplSVData* pSVData = ImplGetSVData(); return pExceptionHandler != nullptr && pSVData->mpApp != nullptr && pSVData->mpDefInst != nullptr; } #ifdef DBG_UTIL namespace vclmain { bool isAlive() { return ImplGetSVData()->mpDefInst; } } #endif bool InitVCL() { if (IsVCLInit()) { SAL_INFO("vcl.app", "Double initialization of vcl"); return true; } if( pExceptionHandler != nullptr ) return false; EmbeddedFontsHelper::clearTemporaryFontFiles(); if( !ImplGetSVData()->mpApp ) { pOwnSvApp = new Application(); } ImplSVData* pSVData = ImplGetSVData(); // remember Main-Thread-Id pSVData->mnMainThreadId = ::osl::Thread::getCurrentIdentifier(); // Initialize Sal pSVData->mpDefInst = CreateSalInstance(); if ( !pSVData->mpDefInst ) return false; pSVData->mpDefInst->AcquireYieldMutex(); // Desktop Environment context (to be able to get value of "system.desktop-environment" as soon as possible) css::uno::setCurrentContext( new DesktopEnvironmentContext( css::uno::getCurrentContext() ) ); // Initialize application instance (should be done after initialization of VCL SAL part) if (pSVData->mpApp) { // call init to initialize application class // soffice/sfx implementation creates the global service manager pSVData->mpApp->Init(); } try { //Now that uno has been bootstrapped we can ask the config what the UI language is so that we can //force that in as $LANGUAGE. That way we can get gtk to render widgets RTL //if we have a RTL UI in an otherwise LTR locale and get gettext using externals (e.g. python) //to match their translations to our preferred UI language OUString aLocaleString(SvtSysLocaleOptions().GetRealUILanguageTag().getGlibcLocaleString(u".UTF-8")); if (!aLocaleString.isEmpty()) { MsLangId::getSystemUILanguage(); //call this now to pin what the system UI really was OUString envVar("LANGUAGE"); osl_setEnvironment(envVar.pData, aLocaleString.pData); } } catch (const uno::Exception &) { TOOLS_INFO_EXCEPTION("vcl.app", "Unable to get ui language:"); } pSVData->mpDefInst->AfterAppInit(); // Fetch AppFileName and make it absolute before the workdir changes... OUString aExeFileName; osl_getExecutableFile( &aExeFileName.pData ); // convert path to native file format OUString aNativeFileName; osl::FileBase::getSystemPathFromFileURL( aExeFileName, aNativeFileName ); pSVData->maAppData.mxAppFileName = aNativeFileName; // Initialize global data pSVData->maGDIData.mxScreenFontList = std::make_shared(); pSVData->maGDIData.mxScreenFontCache = std::make_shared(); pSVData->maGDIData.mxGrfConverter.reset(new GraphicConverter); g_bIsLeanException = getenv("LO_LEAN_EXCEPTION") != nullptr; // Set exception handler pExceptionHandler = osl_addSignalHandler(VCLExceptionSignal_impl, nullptr); #ifndef NDEBUG DbgGUIInitSolarMutexCheck(); #endif #if OSL_DEBUG_LEVEL > 0 DebugEventInjector::getCreate(); #endif #ifndef _WIN32 // Clear startup notification details for child processes // See https://bugs.freedesktop.org/show_bug.cgi?id=11375 for discussion unsetenv("DESKTOP_STARTUP_ID"); #endif return true; } namespace { /** Serves for destroying the VCL UNO wrapper as late as possible. This avoids crash at exit in some special cases when a11y is enabled (e.g., when a bundled extension is registered/deregistered during startup, forcing exit while the app is still in splash screen.) */ class VCLUnoWrapperDeleter : public cppu::WeakImplHelper { virtual void SAL_CALL disposing(lang::EventObject const& rSource) override; }; void VCLUnoWrapperDeleter::disposing(lang::EventObject const& /* rSource */) { ImplSVData* const pSVData = ImplGetSVData(); if (pSVData && pSVData->mpUnoWrapper) { pSVData->mpUnoWrapper->Destroy(); pSVData->mpUnoWrapper = nullptr; } } } void DeInitVCL() { // The LOK Windows map container should be empty assert(vcl::Window::IsLOKWindowsEmpty()); //rhbz#1444437, when using LibreOffice like a library you can't realistically //tear everything down and recreate them on the next call, there's too many //(c++) singletons that point to stuff that gets deleted during shutdown //which won't be recreated on restart. if (comphelper::LibreOfficeKit::isActive()) return; { SolarMutexReleaser r; // unblock threads blocked on that so we can join ::comphelper::JoinAsyncEventNotifiers(); } ImplSVData* pSVData = ImplGetSVData(); // lp#1560328: clear cache before disposing rest of VCL if(pSVData->mpBlendFrameCache) pSVData->mpBlendFrameCache->m_aLastResult.Clear(); pSVData->mbDeInit = true; vcl::DeleteOnDeinitBase::ImplDeleteOnDeInit(); #if OSL_DEBUG_LEVEL > 0 OStringBuffer aBuf( 256 ); aBuf.append( "DeInitVCL: some top Windows are still alive\n" ); tools::Long nTopWindowCount = Application::GetTopWindowCount(); tools::Long nBadTopWindows = nTopWindowCount; for( tools::Long i = 0; i < nTopWindowCount; i++ ) { vcl::Window* pWin = Application::GetTopWindow( i ); // default window will be destroyed further down // but may still be useful during deinit up to that point if( pWin == pSVData->mpDefaultWin ) nBadTopWindows--; else { aBuf.append( "text = \"" ); aBuf.append( OUStringToOString( pWin->GetText(), osl_getThreadTextEncoding() ) ); aBuf.append( "\" type = \"" ); aBuf.append( typeid(*pWin).name() ); aBuf.append( "\", ptr = 0x" ); aBuf.append( reinterpret_cast( pWin ), 16 ); aBuf.append( "\n" ); } } SAL_WARN_IF( nBadTopWindows!=0, "vcl", aBuf.getStr() ); #endif ImageTree::get().shutdown(); osl_removeSignalHandler( pExceptionHandler); pExceptionHandler = nullptr; // free global data pSVData->maGDIData.mxGrfConverter.reset(); pSVData->mpSettingsConfigItem.reset(); // prevent unnecessary painting during Scheduler shutdown // as this processes all pending events in debug builds. ImplGetSystemDependentDataManager().flushAll(); Scheduler::ImplDeInitScheduler(); pSVData->mpWinData->maMsgBoxImgList.clear(); pSVData->maCtrlData.maCheckImgList.clear(); pSVData->maCtrlData.maRadioImgList.clear(); pSVData->maCtrlData.mpDisclosurePlus.reset(); pSVData->maCtrlData.mpDisclosureMinus.reset(); pSVData->mpDefaultWin.disposeAndClear(); #if defined _WIN32 // See GetSystemClipboard (vcl/source/treelist/transfer2.cxx): if (auto const comp = css::uno::Reference( pSVData->m_xSystemClipboard, css::uno::UNO_QUERY)) { SolarMutexReleaser r; // unblock pending "clipboard content changed" notifications comp->dispose(); // will use s_aClipboardSingletonMutex for CWinClipboard } pSVData->m_xSystemClipboard.clear(); #endif #ifndef NDEBUG DbgGUIDeInitSolarMutexCheck(); #endif if ( pSVData->mpUnoWrapper ) { try { uno::Reference const xDesktop = frame::Desktop::create( comphelper::getProcessComponentContext() ); xDesktop->addEventListener(new VCLUnoWrapperDeleter); } catch (uno::Exception const&) { // ignore } } if( pSVData->mpApp || pSVData->maDeInitHook.IsSet() ) { SolarMutexReleaser aReleaser; // call deinit to deinitialize application class // soffice/sfx implementation disposes the global service manager // Warning: After this call you can't call uno services if( pSVData->mpApp ) { pSVData->mpApp->DeInit(); } if( pSVData->maDeInitHook.IsSet() ) { pSVData->maDeInitHook.Call(nullptr); } } if ( pSVData->maAppData.mxSettings ) { if ( pSVData->maAppData.mpCfgListener ) { pSVData->maAppData.mxSettings->GetSysLocale().GetOptions().RemoveListener( pSVData->maAppData.mpCfgListener ); delete pSVData->maAppData.mpCfgListener; } pSVData->maAppData.mxSettings.reset(); } if ( pSVData->maAppData.mpAccelMgr ) { delete pSVData->maAppData.mpAccelMgr; pSVData->maAppData.mpAccelMgr = nullptr; } pSVData->maAppData.maKeyListeners.clear(); pSVData->mpBlendFrameCache.reset(); ImplDeletePrnQueueList(); // destroy all Sal interfaces before destroying the instance // and thereby unloading the plugin pSVData->mpSalSystem.reset(); assert( !pSVData->maSchedCtx.mpSalTimer ); delete pSVData->maSchedCtx.mpSalTimer; pSVData->maSchedCtx.mpSalTimer = nullptr; pSVData->mpDefaultWin = nullptr; pSVData->mpIntroWindow = nullptr; pSVData->maAppData.mpActivePopupMenu = nullptr; pSVData->maAppData.mpWheelWindow = nullptr; pSVData->maGDIData.mpFirstWinGraphics = nullptr; pSVData->maGDIData.mpLastWinGraphics = nullptr; pSVData->maGDIData.mpFirstVirGraphics = nullptr; pSVData->maGDIData.mpLastVirGraphics = nullptr; pSVData->maGDIData.mpFirstPrnGraphics = nullptr; pSVData->maGDIData.mpLastPrnGraphics = nullptr; pSVData->maGDIData.mpFirstVirDev = nullptr; pSVData->maGDIData.mpFirstPrinter = nullptr; pSVData->maFrameData.mpFirstFrame = nullptr; pSVData->maFrameData.mpAppWin = nullptr; pSVData->maFrameData.mpActiveApplicationFrame = nullptr; pSVData->mpWinData->mpCaptureWin = nullptr; pSVData->mpWinData->mpLastDeacWin = nullptr; pSVData->mpWinData->mpFirstFloat = nullptr; pSVData->mpWinData->mpExecuteDialogs.clear(); pSVData->mpWinData->mpExtTextInputWin = nullptr; pSVData->mpWinData->mpTrackWin = nullptr; pSVData->mpWinData->mpAutoScrollWin = nullptr; pSVData->mpWinData->mpLastWheelWindow = nullptr; pSVData->maGDIData.mxScreenFontList.reset(); pSVData->maGDIData.mxScreenFontCache.reset(); // we are iterating over a map and doing erase while inside a loop which is doing erase // hence we can't use clear() here pSVData->maGDIData.maScaleCache.remove_if([](const lru_scale_cache::key_value_pair_t&) { return true; }); pSVData->maGDIData.maThemeDrawCommandsCache.clear(); pSVData->maGDIData.maThemeImageCache.clear(); // Deinit Sal if (pSVData->mpDefInst) { pSVData->mpDefInst->ReleaseYieldMutexAll(); DestroySalInstance( pSVData->mpDefInst ); pSVData->mpDefInst = nullptr; } // This only works on Linux. On Mac and Windows I get very // weird segment violations. #if defined LINUX delete pSVData->mpSalData; #endif if( pOwnSvApp ) { delete pOwnSvApp; pOwnSvApp = nullptr; } EmbeddedFontsHelper::clearTemporaryFontFiles(); } namespace { // only one call is allowed struct WorkerThreadData { oslWorkerFunction pWorker; void * pThreadData; WorkerThreadData( oslWorkerFunction pWorker_, void * pThreadData_ ) : pWorker( pWorker_ ) , pThreadData( pThreadData_ ) { } }; } #ifdef _WIN32 static HANDLE hThreadID = nullptr; static DWORD WINAPI threadmain( _In_ LPVOID pArgs ) { OleInitialize( nullptr ); static_cast(pArgs)->pWorker( static_cast(pArgs)->pThreadData ); delete static_cast(pArgs); OleUninitialize(); hThreadID = nullptr; return 0; } #else static oslThread hThreadID = nullptr; extern "C" { static void MainWorkerFunction( void* pArgs ) { static_cast(pArgs)->pWorker( static_cast(pArgs)->pThreadData ); delete static_cast(pArgs); hThreadID = nullptr; } } // extern "C" #endif void CreateMainLoopThread( oslWorkerFunction pWorker, void * pThreadData ) { #ifdef _WIN32 // sal thread always call CoInitializeEx, so a system dependent implementation is necessary DWORD uThreadID; hThreadID = CreateThread( nullptr, // no security handle 0, // stacksize 0 means default threadmain, // thread worker function new WorkerThreadData( pWorker, pThreadData ), // arguments for worker function 0, // 0 means: create immediately otherwise use CREATE_SUSPENDED &uThreadID ); // thread id to fill #else hThreadID = osl_createThread( MainWorkerFunction, new WorkerThreadData( pWorker, pThreadData ) ); #endif } void JoinMainLoopThread() { if( hThreadID ) { #ifdef _WIN32 WaitForSingleObject(hThreadID, INFINITE); #else osl_joinWithThread(hThreadID); osl_destroyThread( hThreadID ); #endif } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */