b23019c566
This reverts commit 829fa53fe8
, which started all
sorts of `make check` tests to hang for me due to a recursive locking attempt in
> #5 std::mutex::lock (this=0x7f6af81e62c0) at ~/gcc/inst/include/c++/14.0.1/bits/std_mutex.h:113
> #6 std::unique_lock<std::mutex>::lock (this=<synthetic pointer>) at ~/gcc/inst/include/c++/14.0.1/bits/unique_lock.h:147
> #7 std::unique_lock<std::mutex>::unique_lock (this=<synthetic pointer>, __m=...) at ~/gcc/inst/include/c++/14.0.1/bits/unique_lock.h:73
> #8 GrammarCheckingIterator::NextDocId (this=0x7f6af81e6260) at linguistic/source/gciterator.cxx:317
> #9 GrammarCheckingIterator::GetOrCreateDocId (this=this@entry=0x7f6af81e6260, xComponent=uno::Reference to (SwXTextDocument *) 0x7f6af8430ec8) at linguistic/source/gciterator.cxx:339
> #10 0x00007f6b3b9e9da9 in GrammarCheckingIterator::startProofreading (this=0x7f6af81e6260, xDoc=<optimized out>, xIteratorProvider=<optimized out>) at linguistic/source/gciterator.cxx:773
> #11 0x00007f6b16441ec3 in SwDoc::StartGrammarChecking (this=<optimized out>, bSkipStart=bSkipStart@entry=false) at sw/source/core/doc/docnew.cxx:180
> #12 0x00007f6b164c4eae in sw::DocumentTimerManager::DoIdleJobs (this=0x7f6afad005a0) at sw/source/core/doc/DocumentTimerManager.cxx:169
> #13 0x00007f6b3e83a68b in Scheduler::CallbackTaskScheduling () at vcl/source/app/scheduler.cxx:509
> #14 0x00007f6b3eb1725f in SalTimer::CallCallback (this=<optimized out>) at vcl/inc/saltimer.hxx:54
> #15 SvpSalInstance::CheckTimeout (this=this@entry=0xffebe0, bExecuteTimers=bExecuteTimers@entry=true) at vcl/headless/svpinst.cxx:157
> #16 0x00007f6b3eb1761d in SvpSalInstance::ImplYield (this=this@entry=0xffebe0, bWait=bWait@entry=true, bHandleAllCurrentEvents=bHandleAllCurrentEvents@entry=false) at vcl/headless/svpinst.cxx:395
> #17 0x00007f6b3eb17cd5 in SvpSalInstance::DoYield (this=0xffebe0, bWait=<optimized out>, bHandleAllCurrentEvents=<optimized out>) at vcl/headless/svpinst.cxx:467
> #18 0x00007f6b3e86aa44 in ImplYield (i_bWait=true, i_bAllEvents=false) at vcl/source/app/svapp.cxx:394
> #19 0x00007f6b3e86b13b in Application::Execute () at vcl/source/app/svapp.cxx:369
> #20 0x00007f6b3c8e715f in desktop::Desktop::Main (this=0x7ffe5e22fcb0) at desktop/source/app/app.cxx:1615
> #21 0x00007f6b3e87e6fb in ImplSVMain () at vcl/source/app/svmain.cxx:229
> #22 0x00007f6b3e87e9c5 in SVMain () at vcl/source/app/svmain.cxx:261
> #23 0x00007f6b3c91de37 in soffice_main () at desktop/source/app/sofficemain.cxx:93
> #24 0x000000000040078b in sal_main () at desktop/source/app/main.c:51
> #25 main (argc=argc@entry=8, argv=argv@entry=0x7ffe5e22feb8) at desktop/source/app/main.c:49
Change-Id: I60b8b0ba00cae691d6089325e4379a86221dc95b
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/165764
Tested-by: Jenkins
Reviewed-by: Stephan Bergmann <stephan.bergmann@allotropia.de>
219 lines
8.8 KiB
C++
219 lines
8.8 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 .
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <com/sun/star/i18n/XBreakIterator.hpp>
|
|
#include <com/sun/star/lang/XComponent.hpp>
|
|
#include <com/sun/star/lang/XServiceInfo.hpp>
|
|
#include <com/sun/star/lang/XEventListener.hpp>
|
|
#include <com/sun/star/linguistic2/XProofreadingIterator.hpp>
|
|
#include <com/sun/star/linguistic2/XLinguServiceEventListener.hpp>
|
|
#include <com/sun/star/linguistic2/XLinguServiceEventBroadcaster.hpp>
|
|
#include <com/sun/star/util/XChangesBatch.hpp>
|
|
|
|
#include <cppuhelper/implbase.hxx>
|
|
#include <osl/mutex.hxx>
|
|
#include <osl/conditn.hxx>
|
|
#include <osl/thread.h>
|
|
|
|
#include <com/sun/star/uno/Any.hxx>
|
|
#include <comphelper/lok.hxx>
|
|
#include <comphelper/interfacecontainer3.hxx>
|
|
#include <i18nlangtag/lang.h>
|
|
|
|
#include <map>
|
|
#include <optional>
|
|
#include <utility>
|
|
#include <deque>
|
|
|
|
#include "defs.hxx"
|
|
|
|
|
|
struct FPEntry
|
|
{
|
|
// flat paragraph iterator
|
|
css::uno::Reference< css::text::XFlatParagraphIterator > m_xParaIterator;
|
|
|
|
// flat paragraph
|
|
css::uno::Reference< css::text::XFlatParagraph > m_xPara;
|
|
|
|
// document ID to identify different documents
|
|
OUString m_aDocId;
|
|
|
|
// the starting position to be checked
|
|
sal_Int32 m_nStartIndex;
|
|
|
|
// the flag to identify whether the document does automatic grammar checking
|
|
bool m_bAutomatic;
|
|
|
|
FPEntry()
|
|
: m_aDocId()
|
|
, m_nStartIndex( 0 )
|
|
, m_bAutomatic( false )
|
|
{
|
|
}
|
|
};
|
|
|
|
|
|
class GrammarCheckingIterator:
|
|
public cppu::WeakImplHelper
|
|
<
|
|
css::linguistic2::XProofreadingIterator,
|
|
css::linguistic2::XLinguServiceEventListener,
|
|
css::linguistic2::XLinguServiceEventBroadcaster,
|
|
css::lang::XComponent,
|
|
css::lang::XServiceInfo
|
|
>,
|
|
public LinguDispatcher,
|
|
public comphelper::LibreOfficeKit::ThreadJoinable
|
|
{
|
|
//the queue is keeping track of all sentences to be checked
|
|
//every element of this queue is a FlatParagraphEntry struct-object
|
|
typedef std::deque< FPEntry > FPQueue_t;
|
|
|
|
// queue for entries to be processed
|
|
FPQueue_t m_aFPEntriesQueue;
|
|
|
|
// the flag to end the endless loop
|
|
bool m_bEnd;
|
|
|
|
// Note that it must be the pointer and not the uno-reference to check if it is the same implementation object
|
|
typedef std::map< XComponent *, OUString > DocMap_t;
|
|
DocMap_t m_aDocIdMap;
|
|
|
|
|
|
// BCP-47 language tag -> implname mapping
|
|
typedef std::map< OUString, OUString > GCImplNames_t;
|
|
GCImplNames_t m_aGCImplNamesByLang;
|
|
|
|
// implname -> UNO reference mapping
|
|
typedef std::map< OUString, css::uno::Reference< css::linguistic2::XProofreader > > GCReferences_t;
|
|
GCReferences_t m_aGCReferencesByService;
|
|
|
|
OUString m_aCurCheckedDocId;
|
|
bool m_bGCServicesChecked;
|
|
sal_Int32 m_nDocIdCounter;
|
|
osl::Condition m_aWakeUpThread;
|
|
oslThread m_thread;
|
|
|
|
//! beware of initialization order!
|
|
static osl::Mutex& MyMutex();
|
|
comphelper::OInterfaceContainerHelper3<css::lang::XEventListener> m_aEventListeners;
|
|
comphelper::OInterfaceContainerHelper3<css::linguistic2::XLinguServiceEventListener> m_aNotifyListeners;
|
|
|
|
css::uno::Reference< css::i18n::XBreakIterator > m_xBreakIterator;
|
|
mutable css::uno::Reference< css::util::XChangesBatch > m_xUpdateAccess;
|
|
|
|
void TerminateThread();
|
|
|
|
sal_Int32 NextDocId();
|
|
OUString GetOrCreateDocId( const css::uno::Reference< css::lang::XComponent > &xComp );
|
|
|
|
void AddEntry(
|
|
const css::uno::Reference< css::text::XFlatParagraphIterator >& xFlatParaIterator,
|
|
const css::uno::Reference< css::text::XFlatParagraph >& xFlatPara,
|
|
const OUString &rDocId, sal_Int32 nStartIndex, bool bAutomatic );
|
|
|
|
void ProcessResult( const css::linguistic2::ProofreadingResult &rRes,
|
|
const css::uno::Reference< css::text::XFlatParagraphIterator > &rxFlatParagraphIterator,
|
|
bool bIsAutomaticChecking );
|
|
|
|
sal_Int32 GetSuggestedEndOfSentence( const OUString &rText, sal_Int32 nSentenceStartPos, const css::lang::Locale &rLocale );
|
|
|
|
void GetConfiguredGCSvcs_Impl();
|
|
css::uno::Reference< css::linguistic2::XProofreader > GetGrammarChecker( css::lang::Locale & rLocale );
|
|
|
|
css::uno::Reference< css::util::XChangesBatch > const & GetUpdateAccess() const;
|
|
|
|
GrammarCheckingIterator( const GrammarCheckingIterator & ) = delete;
|
|
GrammarCheckingIterator & operator = ( const GrammarCheckingIterator & ) = delete;
|
|
|
|
// Gets the grammar checker service, using fallback locales if necessary,
|
|
// and the BCP-47 tag for the updated locale, if the fallback was used.
|
|
// Precondition: MyMutex() is locked.
|
|
std::pair<OUString, std::optional<OUString>>
|
|
getServiceForLocale(const css::lang::Locale& rLocale) const;
|
|
|
|
public:
|
|
|
|
void DequeueAndCheck();
|
|
|
|
explicit GrammarCheckingIterator();
|
|
virtual ~GrammarCheckingIterator() override;
|
|
|
|
// XProofreadingIterator
|
|
virtual void SAL_CALL startProofreading( const css::uno::Reference< css::uno::XInterface >& xDocument, const css::uno::Reference< css::text::XFlatParagraphIteratorProvider >& xIteratorProvider ) override;
|
|
virtual css::linguistic2::ProofreadingResult SAL_CALL checkSentenceAtPosition( const css::uno::Reference< css::uno::XInterface >& xDocument, const css::uno::Reference< css::text::XFlatParagraph >& xFlatParagraph, const OUString& aText, const css::lang::Locale& aLocale, ::sal_Int32 nStartOfSentencePosition, ::sal_Int32 nSuggestedBehindEndOfSentencePosition, ::sal_Int32 nErrorPositionInParagraph ) override;
|
|
virtual void SAL_CALL resetIgnoreRules( ) override;
|
|
virtual sal_Bool SAL_CALL isProofreading( const css::uno::Reference< css::uno::XInterface >& xDocument ) override;
|
|
|
|
// XLinguServiceEventListener
|
|
virtual void SAL_CALL processLinguServiceEvent( const css::linguistic2::LinguServiceEvent& aLngSvcEvent ) override;
|
|
|
|
// XLinguServiceEventBroadcaster
|
|
virtual sal_Bool SAL_CALL addLinguServiceEventListener( const css::uno::Reference< css::linguistic2::XLinguServiceEventListener >& xLstnr ) override;
|
|
virtual sal_Bool SAL_CALL removeLinguServiceEventListener( const css::uno::Reference< css::linguistic2::XLinguServiceEventListener >& xLstnr ) override;
|
|
|
|
// XComponent
|
|
virtual void SAL_CALL dispose( ) override;
|
|
virtual void SAL_CALL addEventListener( const css::uno::Reference< css::lang::XEventListener >& xListener ) override;
|
|
virtual void SAL_CALL removeEventListener( const css::uno::Reference< css::lang::XEventListener >& aListener ) override;
|
|
|
|
// XEventListener
|
|
virtual void SAL_CALL disposing( const css::lang::EventObject& Source ) override;
|
|
|
|
// XServiceInfo
|
|
virtual OUString SAL_CALL getImplementationName( ) override;
|
|
virtual sal_Bool SAL_CALL supportsService( const OUString& ServiceName ) override;
|
|
virtual css::uno::Sequence< OUString > SAL_CALL getSupportedServiceNames( ) override;
|
|
|
|
// LinguDispatcher
|
|
virtual void SetServiceList( const css::lang::Locale &rLocale, const css::uno::Sequence< OUString > &rSvcImplNames ) override;
|
|
virtual css::uno::Sequence< OUString > GetServiceList( const css::lang::Locale &rLocale ) const override;
|
|
|
|
// comphelper::LibreOfficeKit::ThreadJoinable
|
|
virtual bool joinThreads() override;
|
|
};
|
|
|
|
|
|
/** Implementation of the css::container::XStringKeyMap interface
|
|
*/
|
|
class LngXStringKeyMap : public ::cppu::WeakImplHelper<css::container::XStringKeyMap>
|
|
{
|
|
public:
|
|
LngXStringKeyMap();
|
|
|
|
virtual css::uno::Any SAL_CALL getValue(const OUString& aKey) override;
|
|
virtual sal_Bool SAL_CALL hasValue(const OUString& aKey) override;
|
|
virtual void SAL_CALL insertValue(const OUString& aKey, const css::uno::Any& aValue) override;
|
|
virtual ::sal_Int32 SAL_CALL getCount() override;
|
|
virtual OUString SAL_CALL getKeyByIndex(::sal_Int32 nIndex) override;
|
|
virtual css::uno::Any SAL_CALL getValueByIndex(::sal_Int32 nIndex) override;
|
|
|
|
private:
|
|
LngXStringKeyMap(LngXStringKeyMap const &) = delete;
|
|
void operator=(LngXStringKeyMap const &) = delete;
|
|
|
|
~LngXStringKeyMap() override{};
|
|
|
|
std::map<OUString, css::uno::Any> maMap;
|
|
};
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|