1
0
Fork 0
libreoffice/vcl/source/app/svmain.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

716 lines
21 KiB
C++

/* -*- 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 <sal/config.h>
#include <sal/log.hxx>
#include <cassert>
#include <osl/file.hxx>
#include <osl/signal.h>
#include <desktop/exithelper.h>
#include <comphelper/accessibleeventnotifier.hxx>
#include <comphelper/processfactory.hxx>
#include <comphelper/asyncnotification.hxx>
#include <i18nlangtag/mslangid.hxx>
#include <unotools/syslocale.hxx>
#include <unotools/syslocaleoptions.hxx>
#include <utility>
#include <vcl/QueueInfo.hxx>
#include <vcl/svapp.hxx>
#include <vcl/vclmain.hxx>
#include <vcl/wrkwin.hxx>
#include <vcl/cvtgrf.hxx>
#include <vcl/scheduler.hxx>
#include <vcl/image.hxx>
#include <vcl/ImageTree.hxx>
#include <vcl/settings.hxx>
#include <vcl/toolkit/unowrap.hxx>
#include <configsettings.hxx>
#include <tools/lazydelete.hxx>
#include <vcl/embeddedfontshelper.hxx>
#include <vcl/toolkit/dialog.hxx>
#include <vcl/menu.hxx>
#include <vcl/virdev.hxx>
#include <vcl/print.hxx>
#include <debugevent.hxx>
#include <scrwnd.hxx>
#include <windowdev.hxx>
#include <svdata.hxx>
#ifdef _WIN32
#include <svsys.h>
#include <process.h>
#include <ole2.h>
#else
#include <stdlib.h>
#endif
#ifdef ANDROID
#include <cppuhelper/bootstrap.hxx>
#include <jni.h>
#endif
#include <impfontcache.hxx>
#include <salinst.hxx>
#include <vcl/svmain.hxx>
#include <dbggui.hxx>
#include <accmgr.hxx>
#include <font/PhysicalFontCollection.hxx>
#include <print.h>
#include <salsys.hxx>
#include <saltimer.hxx>
#include <displayconnectiondispatch.hxx>
#include <config_features.h>
#include <config_feature_opencl.h>
#include <systools/opensslinit.hxx>
#include <osl/process.h>
#include <com/sun/star/lang/XComponent.hpp>
#include <com/sun/star/frame/Desktop.hpp>
#ifdef _WIN32
#include <com/sun/star/datatransfer/clipboard/XClipboard.hpp>
#endif
#include <comphelper/lok.hxx>
#include <cppuhelper/implbase.hxx>
#include <uno/current_context.hxx>
#include <opencl/OpenCLZone.hxx>
#include <opengl/zone.hxx>
#include <skia/zone.hxx>
#include <watchdog.hxx>
#include <basegfx/utils/systemdependentdata.hxx>
#include <comphelper/diagnose_ex.hxx>
#if OSL_DEBUG_LEVEL > 0
#include <typeinfo>
#include <rtl/strbuf.hxx>
#endif
#ifdef LINUX
#include <unx/gendata.hxx>
#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();
#if !defined(_WIN32) && !defined(SYSTEM_OPENSSL)
if (!bWasInitVCL)
{
static constexpr OUString name(u"SSL_CERT_FILE"_ustr);
OUString temp;
if (osl_getEnvironment(name.pData, &temp.pData) == osl_Process_E_NotFound)
{
// Try to point bundled OpenSSL to some system certificate file
// ... this only works if the client actually calls
// SSL_CTX_set_default_verify_paths() or similar; e.g. python ssl.
char const*const path = GetCABundleFile();
if (path == nullptr) {
SAL_WARN("vcl", "no OpenSSL CA certificate bundle found");
} else {
OUString const filepath(::rtl::OStringToOUString(
::std::string_view(path), osl_getThreadTextEncoding()));
osl_setEnvironment(name.pData, filepath.pData);
}
}
}
#endif
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();
}
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( css::uno::Reference< css::uno::XCurrentContext > ctx)
: m_xNextContext(std::move( 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(u"LANGUAGE"_ustr);
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<vcl::font::PhysicalFontCollection>();
pSVData->maGDIData.mxScreenFontCache = std::make_shared<ImplFontCache>();
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<css::lang::XEventListener>
{
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();
// 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();
}
// lp#1560328: clear cache before disposing rest of VCL
if(pSVData->mpBlendFrameCache)
pSVData->mpBlendFrameCache->m_aLastResult.Clear();
pSVData->mbDeInit = true;
// Some events may need to access objects destroyed in ImplDeleteOnDeInit, so process them first
Scheduler::ProcessEventsToIdle();
tools::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 = \""
+ OUStringToOString( pWin->GetText(), osl_getThreadTextEncoding() )
+ "\" type = \""
+ typeid(*pWin).name()
+ "\", ptr = 0x"
+ OString::number(reinterpret_cast<sal_Int64>( pWin ), 16 )
+ "\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();
#if defined _WIN32
// See GetSystemClipboard (vcl/source/treelist/transfer2.cxx):
if (auto const comp = css::uno::Reference<css::lang::XComponent>(
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
Scheduler::ImplDeInitScheduler();
pSVData->mpWinData->maMsgBoxImgList.clear();
pSVData->maCtrlData.maCheckImgList.clear();
pSVData->maCtrlData.maRadioImgList.clear();
pSVData->maCtrlData.moDisclosurePlus.reset();
pSVData->maCtrlData.moDisclosureMinus.reset();
pSVData->mpDefaultWin.disposeAndClear();
#ifndef NDEBUG
DbgGUIDeInitSolarMutexCheck();
#endif
if ( pSVData->mpUnoWrapper )
{
try
{
uno::Reference<frame::XDesktop2> 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();
pSVData->dropCaches();
comphelper::AccessibleEventNotifier::shutdown();
// Deinit Sal
if (pSVData->mpDefInst)
{
pSVData->mpDefInst->ReleaseYieldMutex(true);
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 unsigned __stdcall threadmain(void* pArgs)
{
OleInitialize( nullptr );
static_cast<WorkerThreadData*>(pArgs)->pWorker( static_cast<WorkerThreadData*>(pArgs)->pThreadData );
delete static_cast<WorkerThreadData*>(pArgs);
OleUninitialize();
hThreadID = nullptr;
return 0;
}
#else
static oslThread hThreadID = nullptr;
extern "C"
{
static void MainWorkerFunction( void* pArgs )
{
static_cast<WorkerThreadData*>(pArgs)->pWorker( static_cast<WorkerThreadData*>(pArgs)->pThreadData );
delete static_cast<WorkerThreadData*>(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
hThreadID = reinterpret_cast<HANDLE>(_beginthreadex(
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
nullptr )); // 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: */