PDF4QT/Pdf4QtLib/sources/pdfexecutionpolicy.h

188 lines
6.0 KiB
C
Raw Normal View History

2021-04-30 20:12:10 +02:00
// Copyright (C) 2020-2021 Jakub Melka
2020-01-18 14:55:26 +01:00
//
2020-12-20 19:03:58 +01:00
// This file is part of Pdf4Qt.
2020-01-18 14:55:26 +01:00
//
2020-12-20 19:03:58 +01:00
// Pdf4Qt is free software: you can redistribute it and/or modify
2020-01-18 14:55:26 +01:00
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
2021-04-30 20:12:10 +02:00
// with the written consent of the copyright owner, any later version.
2020-01-18 14:55:26 +01:00
//
2020-12-20 19:03:58 +01:00
// Pdf4Qt is distributed in the hope that it will be useful,
2020-01-18 14:55:26 +01:00
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
2020-12-20 19:03:58 +01:00
// along with Pdf4Qt. If not, see <https://www.gnu.org/licenses/>.
2020-01-18 14:55:26 +01:00
#ifndef PDFEXECUTIONPOLICY_H
#define PDFEXECUTIONPOLICY_H
#include "pdfglobal.h"
#include <QSemaphore>
#include <QThreadPool>
2020-01-18 14:55:26 +01:00
#include <atomic>
#include <execution>
namespace pdf
{
struct PDFExecutionPolicyHolder;
/// Defines thread execution policy based on settings and actual number of page content
/// streams being processed. It can regulate number of threads executed at each
/// point, where execution policy is used.
2020-12-20 19:03:58 +01:00
class Pdf4QtLIBSHARED_EXPORT PDFExecutionPolicy
2020-01-18 14:55:26 +01:00
{
public:
enum class Scope
{
Page, ///< Used, when we are processing page objects
Content, ///< Used, when we are processing objects from page content streams
Unknown ///< Unknown scope, usually tries to parallelize
};
enum class Strategy
{
SingleThreaded,
PageMultithreaded,
AlwaysMultithreaded
};
/// Sets multithreading strategy
/// \param strategy Strategy
static void setStrategy(Strategy strategy);
/// Determines, if we should parallelize for scope
/// \param scope Scope for which we want to determine execution policy
2020-01-18 14:55:26 +01:00
static bool isParallelizing(Scope scope);
template<typename ForwardIt, typename UnaryFunction>
class Runnable : public QRunnable
{
public:
2021-06-02 19:26:47 +02:00
explicit inline Runnable(ForwardIt it, ForwardIt itEnd, UnaryFunction* function, QSemaphore* semaphore) :
m_forwardIt(qMove(it)),
2021-06-02 19:26:47 +02:00
m_forwardItEnd(qMove(itEnd)),
m_function(function),
2021-06-02 19:26:47 +02:00
m_semaphore(semaphore),
m_packSize(static_cast<int>(std::distance(m_forwardIt, m_forwardItEnd)))
{
setAutoDelete(true);
}
virtual void run() override
{
2021-06-02 19:26:47 +02:00
QSemaphoreReleaser semaphoreReleaser(m_semaphore, m_packSize);
for (auto it = m_forwardIt; it != m_forwardItEnd; ++it)
{
(*m_function)(*it);
}
}
private:
ForwardIt m_forwardIt;
2021-06-02 19:26:47 +02:00
ForwardIt m_forwardItEnd;
UnaryFunction* m_function;
QSemaphore* m_semaphore;
2021-06-02 19:26:47 +02:00
int m_packSize;
};
2020-01-18 14:55:26 +01:00
template<typename ForwardIt, typename UnaryFunction>
static void execute(Scope scope, ForwardIt first, ForwardIt last, UnaryFunction f)
{
if (isParallelizing(scope))
{
QSemaphore semaphore(0);
int count = static_cast<int>(std::distance(first, last));
2021-06-02 19:26:47 +02:00
int remainder = count;
int bucketSize = 1;
// For page scope, we do not divide the tasks into buckets, i.e.
// each bucket will have size 1. But if we are in a content scope,
// then we are processing smaller task, so we divide the work
// into buckets of appropriate size.
if (scope != Scope::Page)
{
const int buckets = 32 * QThread::idealThreadCount();
bucketSize = qMax(1, count / buckets);
}
QThreadPool* pool = getThreadPool(scope);
2021-06-02 19:26:47 +02:00
// Divide tasks into buckets with given bucket size
auto it = first;
while (remainder > 0)
{
2021-06-02 19:26:47 +02:00
const int currentSize = qMin(remainder, bucketSize);
auto itStart = it;
auto itEnd = std::next(it, currentSize);
pool->start(new Runnable(itStart, itEnd, &f, &semaphore));
remainder -= currentSize;
std::advance(it, currentSize);
}
2021-06-02 19:26:47 +02:00
Q_ASSERT(it == last);
semaphore.acquire(count);
2020-01-18 14:55:26 +01:00
}
else
{
std::for_each(std::execution::sequenced_policy(), first, last, f);
}
}
template<typename ForwardIt, typename Comparator>
static void sort(Scope scope, ForwardIt first, ForwardIt last, Comparator f)
{
Q_UNUSED(scope);
// We always sort by single thread
std::sort(std::execution::sequenced_policy(), first, last, f);
2020-01-18 14:55:26 +01:00
}
/// Returns number of active threads for given scope
static int getActiveThreadCount(Scope scope);
/// Returns maximal number of threads for given scope
static int getMaxThreadCount(Scope scope);
/// Sets maximal number of threads for given scope
static void setMaxThreadCount(Scope scope, int count);
/// Returns ideal thread count for given scope
static int getIdealThreadCount(Scope scope);
/// Returns number of currently processed content streams
static int getContentStreamCount();
2020-01-18 14:55:26 +01:00
/// Starts processing content stream
static void startProcessingContentStream();
/// Ends processing content stream
static void endProcessingContentStream();
/// Finalize multithreading - must be called at the end of program
static void finalize();
2020-01-18 14:55:26 +01:00
private:
friend struct PDFExecutionPolicyHolder;
/// Returns thread pool based on scope
static QThreadPool* getThreadPool(Scope scope);
2020-01-18 14:55:26 +01:00
explicit PDFExecutionPolicy();
std::atomic<int> m_contentStreamsCount;
2020-01-18 14:55:26 +01:00
std::atomic<Strategy> m_strategy;
};
} // namespace pdf
#endif // PDFEXECUTIONPOLICY_H