2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
|
|
|
|
#include "trenderer.h"
|
|
|
|
#include "trendererP.h"
|
|
|
|
|
|
|
|
// TnzCore includes
|
|
|
|
#include "tsystem.h"
|
|
|
|
#include "tthreadmessage.h"
|
|
|
|
#include "tthread.h"
|
|
|
|
#include "tconvert.h"
|
|
|
|
#include "tstream.h"
|
|
|
|
#include "trasterimage.h"
|
|
|
|
#include "trop.h"
|
|
|
|
#include "timagecache.h"
|
|
|
|
#include "tstopwatch.h"
|
2020-10-08 10:27:55 +13:00
|
|
|
#include "timage_io.h"
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
// TnzBase includes
|
|
|
|
#include "trenderresourcemanager.h"
|
|
|
|
#include "tpredictivecachemanager.h"
|
|
|
|
|
|
|
|
// Qt includes
|
|
|
|
#include <QEventLoop>
|
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QReadWriteLock>
|
|
|
|
#include <QReadLocker>
|
|
|
|
#include <QWriteLocker>
|
|
|
|
#include <QThreadStorage>
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// Debug
|
2016-03-19 06:57:51 +13:00
|
|
|
//#define DIAGNOSTICS
|
|
|
|
//#include "diagnostics.h"
|
|
|
|
|
|
|
|
#include <queue>
|
|
|
|
#include <functional>
|
|
|
|
|
2018-01-12 20:00:36 +13:00
|
|
|
#include <QOffscreenSurface>
|
|
|
|
#include <QSurfaceFormat>
|
|
|
|
#include <QOpenGLContext>
|
|
|
|
#include <QThread>
|
|
|
|
#include <QGuiApplication>
|
|
|
|
|
2016-03-19 06:57:51 +13:00
|
|
|
using namespace TThread;
|
|
|
|
|
|
|
|
std::vector<const TFx *> calculateSortedFxs(TRasterFxP rootFx);
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
// Preliminaries - Anonymous namespace
|
|
|
|
//================================================================================
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
namespace {
|
|
|
|
// TRenderer-thread association storage. It provides TRenderers the per-thread
|
|
|
|
// singleton status from inside a rendering process.
|
2016-03-19 06:57:51 +13:00
|
|
|
QThreadStorage<TRendererImp **> rendererStorage;
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// Same for render process ids.
|
2016-03-19 06:57:51 +13:00
|
|
|
QThreadStorage<unsigned long *> renderIdsStorage;
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// Interlacing functions for field-based rendering
|
|
|
|
inline void interlace(TRasterP f0, const TRasterP &f1, int field) {
|
|
|
|
if (f0->getPixelSize() != f1->getPixelSize())
|
|
|
|
throw TException("interlace: invalid raster combination");
|
|
|
|
|
|
|
|
assert(f0->getBounds() == f1->getBounds());
|
|
|
|
|
|
|
|
int outWrapBytes = f0->getWrap() * f0->getPixelSize();
|
|
|
|
int inWrapBytes = f1->getWrap() * f1->getPixelSize();
|
|
|
|
int outWrapBytes2 = outWrapBytes * 2;
|
|
|
|
int inWrapBytes2 = inWrapBytes * 2;
|
|
|
|
int rowSize = f0->getLx() * f0->getPixelSize();
|
|
|
|
|
|
|
|
f1->lock();
|
|
|
|
f0->lock();
|
|
|
|
UCHAR *rowIn = f1->getRawData();
|
|
|
|
UCHAR *rowOut = f0->getRawData();
|
|
|
|
|
|
|
|
if (field) rowIn += inWrapBytes;
|
|
|
|
|
|
|
|
int count = f0->getLy() / 2;
|
|
|
|
while (--count) {
|
|
|
|
memcpy(rowOut, rowIn, rowSize);
|
|
|
|
rowIn += inWrapBytes2;
|
|
|
|
rowOut += outWrapBytes2;
|
|
|
|
}
|
|
|
|
|
|
|
|
f1->unlock();
|
|
|
|
f0->unlock();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
2016-06-15 18:43:10 +12:00
|
|
|
} // anonymous namespace
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
// Preliminaries - output rasters management
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//===================
|
|
|
|
// RasterItem
|
|
|
|
//-------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! The RasterItem class represents a reusable output raster for rendering
|
|
|
|
//! purposes.
|
|
|
|
//! RasterItems are used as TRenderer's output rasters in order to avoid the
|
|
|
|
//! overhead
|
2016-03-19 06:57:51 +13:00
|
|
|
//! of allocating one raster for each rendered frame.
|
2016-06-15 18:43:10 +12:00
|
|
|
//! Each frame-rendering task will lock a RasterItem as preallocated output
|
|
|
|
//! before starting the
|
2016-03-19 06:57:51 +13:00
|
|
|
//! render, therefore changing the \b RasterItem::m_busy attribute accordingly.
|
2016-06-15 18:43:10 +12:00
|
|
|
//! As each frame is rendered on a separate thread, the number of RasterItems
|
|
|
|
//! that
|
|
|
|
//! TRenderer will allocate depends on the number of rendering threads specified
|
|
|
|
//! to it.
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//! \sa RasterPool, TRenderer classes
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
class RasterItem {
|
|
|
|
std::string m_rasterId;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
public:
|
2016-06-15 18:43:10 +12:00
|
|
|
int m_bpp;
|
|
|
|
bool m_busy;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//---------------------------------------------------------
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
RasterItem(const TDimension &size, int bpp, bool busyFlag)
|
|
|
|
: m_rasterId(""), m_bpp(bpp), m_busy(busyFlag) {
|
|
|
|
TRasterP raster;
|
|
|
|
if (bpp == 32)
|
|
|
|
raster = TRaster32P(size);
|
|
|
|
else if (bpp == 64)
|
|
|
|
raster = TRaster64P(size);
|
|
|
|
else
|
|
|
|
assert(false);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
m_rasterId = TImageCache::instance()->getUniqueId();
|
|
|
|
TImageCache::instance()->add(m_rasterId, TRasterImageP(raster));
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//---------------------------------------------------------
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
~RasterItem() { TImageCache::instance()->remove(m_rasterId); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//---------------------------------------------------------
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRasterP getRaster() const {
|
|
|
|
TRasterImageP rimg =
|
|
|
|
(TRasterImageP)TImageCache::instance()->get(m_rasterId, true);
|
|
|
|
return rimg ? rimg->getRaster() : TRasterP();
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
private:
|
2016-06-15 18:43:10 +12:00
|
|
|
// not implemented
|
|
|
|
RasterItem();
|
|
|
|
RasterItem(const TRaster &RasterItem);
|
2016-03-19 06:57:51 +13:00
|
|
|
};
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//===================
|
|
|
|
// RasterPool
|
|
|
|
//-------------------
|
|
|
|
|
|
|
|
//! Stores a list of RasterItems under TRenderer's requests.
|
2016-06-15 18:43:10 +12:00
|
|
|
class RasterPool {
|
|
|
|
TDimension m_size;
|
|
|
|
int m_bpp;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
typedef std::list<RasterItem *> RasterRepository;
|
|
|
|
RasterRepository m_rasterRepository;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TThread::Mutex m_repositoryLock;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
public:
|
2016-06-15 18:43:10 +12:00
|
|
|
RasterPool() : m_size(-1, -1) {}
|
|
|
|
~RasterPool();
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void setRasterSpecs(const TDimension &size, int bpp);
|
|
|
|
void getRasterSpecs(TDimension &size, int &bpp) {
|
|
|
|
size = m_size;
|
|
|
|
bpp = m_bpp;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRasterP getRaster();
|
|
|
|
TRasterP getRaster(const TDimension &size, int bpp);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void releaseRaster(const TRasterP &r);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void clear();
|
2016-03-19 06:57:51 +13:00
|
|
|
};
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RasterPool::clear() {
|
|
|
|
QMutexLocker sl(&m_repositoryLock);
|
|
|
|
clearPointerContainer(m_rasterRepository);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RasterPool::setRasterSpecs(const TDimension &size, int bpp) {
|
|
|
|
if (size != m_size || bpp != m_bpp) {
|
|
|
|
m_size = size;
|
|
|
|
m_bpp = bpp;
|
|
|
|
clear();
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRasterP RasterPool::getRaster(const TDimension &size, int bpp) {
|
|
|
|
assert(size == m_size && bpp == m_bpp);
|
|
|
|
return getRaster();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! Returns for the first not-busy raster
|
|
|
|
TRasterP RasterPool::getRaster() {
|
|
|
|
QMutexLocker sl(&m_repositoryLock);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
RasterRepository::iterator it = m_rasterRepository.begin();
|
|
|
|
while (it != m_rasterRepository.end()) {
|
|
|
|
RasterItem *rasItem = *it;
|
|
|
|
if (rasItem->m_busy == false) {
|
|
|
|
TRasterP raster = rasItem->getRaster();
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
if (!raster) {
|
|
|
|
delete rasItem;
|
|
|
|
m_rasterRepository.erase(it++);
|
|
|
|
continue;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
rasItem->m_busy = true;
|
|
|
|
raster->clear();
|
|
|
|
return raster;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
++it;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
RasterItem *rasItem = new RasterItem(m_size, m_bpp, true);
|
|
|
|
m_rasterRepository.push_back(rasItem);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
return rasItem->getRaster();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! Cerca il raster \b r in m_rasterRepository; se lo trova setta a \b false il
|
|
|
|
//! campo \b m_busy.
|
|
|
|
void RasterPool::releaseRaster(const TRasterP &r) {
|
|
|
|
if (!r) return;
|
|
|
|
|
|
|
|
QMutexLocker sl(&m_repositoryLock);
|
|
|
|
for (RasterRepository::iterator it = m_rasterRepository.begin();
|
|
|
|
it != m_rasterRepository.end(); ++it) {
|
|
|
|
RasterItem *rasItem = *it;
|
|
|
|
if (rasItem->getRaster()->getRawData() == r->getRawData()) {
|
|
|
|
assert(rasItem->m_busy);
|
|
|
|
rasItem->m_busy = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
RasterPool::~RasterPool() {
|
|
|
|
/*if (m_rasterRepository.size())
|
|
|
|
TSystem::outputDebug("~RasterPool: itemCount = " + toString
|
|
|
|
((int)m_rasterRepository.size())+" (should be 0)\n");*/
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// Release all raster items
|
|
|
|
clear();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
// Internal rendering classes declaration
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//=====================
|
|
|
|
// TRendererImp
|
|
|
|
//---------------------
|
|
|
|
|
2016-06-29 18:17:12 +12:00
|
|
|
class TRendererImp final : public TSmartObject {
|
2016-03-19 06:57:51 +13:00
|
|
|
public:
|
2016-06-15 18:43:10 +12:00
|
|
|
struct RenderInstanceInfos {
|
|
|
|
int m_canceled;
|
|
|
|
int m_activeTasks;
|
|
|
|
int m_status;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
RenderInstanceInfos()
|
|
|
|
: m_canceled(false), m_activeTasks(0), m_status(TRenderer::IDLE) {}
|
|
|
|
};
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
public:
|
2016-06-15 18:43:10 +12:00
|
|
|
typedef std::vector<TRenderPort *> PortContainer;
|
|
|
|
typedef PortContainer::iterator PortContainerIterator;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
QReadWriteLock m_portsLock;
|
|
|
|
PortContainer m_ports;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
QMutex m_renderInstancesMutex;
|
|
|
|
std::map<unsigned long, RenderInstanceInfos> m_activeInstances;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! The renderer Id is a number uniquely associated with a TRenderer instance.
|
|
|
|
static unsigned long m_rendererIdCounter;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! The render Id is a number uniquely associated with a render process
|
|
|
|
//! started
|
|
|
|
//! with the startRendering() method.
|
|
|
|
static unsigned long m_renderIdCounter;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long m_rendererId;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
Executor m_executor;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool m_precomputingEnabled;
|
|
|
|
RasterPool m_rasterPool;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
std::vector<TRenderResourceManager *> m_managers;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TAtomicVar m_undoneTasks;
|
|
|
|
// std::vector<QEventLoop*> m_waitingLoops;
|
|
|
|
std::vector<bool *> m_waitingLoops;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRasterFxP rootFx;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//-------------------------------------------------------------
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRendererImp(int nThreads);
|
|
|
|
~TRendererImp();
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void startRendering(unsigned long renderId,
|
|
|
|
const std::vector<TRenderer::RenderData> &renderDatas);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void notifyRasterStarted(const TRenderPort::RenderData &rd);
|
|
|
|
void notifyRasterCompleted(const TRenderPort::RenderData &rd);
|
|
|
|
void notifyRasterFailure(const TRenderPort::RenderData &rd, TException &e);
|
|
|
|
void notifyRenderFinished(bool isCanceled = false);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void addPort(TRenderPort *port);
|
|
|
|
void removePort(TRenderPort *port);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void abortRendering(unsigned long renderId);
|
|
|
|
void stopRendering(bool waitForCompleteStop);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool hasToDie(unsigned long renderId);
|
|
|
|
int getRenderStatus(unsigned long renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void enablePrecomputing(bool on) { m_precomputingEnabled = on; }
|
|
|
|
bool isPrecomputingEnabled() const { return m_precomputingEnabled; }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void setThreadsCount(int nThreads) { m_executor.setMaxActiveTasks(nThreads); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void declareRenderStart(unsigned long renderId);
|
|
|
|
inline void declareRenderEnd(unsigned long renderId);
|
|
|
|
inline void declareFrameStart(double frame);
|
|
|
|
inline void declareFrameEnd(double frame);
|
|
|
|
inline void declareStatusStart(int renderStatus);
|
|
|
|
inline void declareStatusEnd(int renderStatus);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void quitWaitingLoops();
|
2016-03-19 06:57:51 +13:00
|
|
|
};
|
|
|
|
|
2016-04-15 17:11:23 +12:00
|
|
|
#ifdef _WIN32
|
2016-03-19 06:57:51 +13:00
|
|
|
template class DVAPI TSmartPointerT<TRendererImp>;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef TSmartPointerT<TRendererImp> TRendererImpP;
|
|
|
|
|
|
|
|
unsigned long TRendererImp::m_rendererIdCounter = 0;
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRendererImp::m_renderIdCounter = 0;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//===================
|
|
|
|
// RenderTask
|
|
|
|
//-------------------
|
|
|
|
|
2016-06-29 18:17:12 +12:00
|
|
|
class RenderTask final : public TThread::Runnable {
|
2016-06-15 18:43:10 +12:00
|
|
|
std::vector<double> m_frames;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long m_taskId;
|
|
|
|
unsigned long m_renderId;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRendererImpP m_rendererImp;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TFxPair m_fx;
|
|
|
|
TPointD m_framePos;
|
|
|
|
TDimension m_frameSize;
|
|
|
|
TRenderSettings m_info;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool m_fieldRender, m_stereoscopic;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
Mutex m_rasterGuard;
|
|
|
|
TTile m_tileA; // in normal and field rendering, Rendered at given frame; in
|
|
|
|
// stereoscopic, rendered left frame
|
|
|
|
TTile m_tileB; // in field rendering, rendered at frame + 0.5; in
|
|
|
|
// stereoscopic, rendered right frame
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
public:
|
2016-06-15 18:43:10 +12:00
|
|
|
RenderTask(unsigned long renderId, unsigned long taskId, double frame,
|
|
|
|
const TRenderSettings &ri, const TFxPair &fx,
|
|
|
|
const TPointD &framePos, const TDimension &frameSize,
|
|
|
|
const TRendererImpP &rendererImp);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
~RenderTask() {}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void addFrame(double frame) { m_frames.push_back(frame); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void buildTile(TTile &tile);
|
|
|
|
void releaseTiles();
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void onFrameStarted();
|
|
|
|
void onFrameCompleted();
|
|
|
|
void onFrameFailed(TException &e);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void preRun();
|
2016-06-19 20:06:29 +12:00
|
|
|
void run() override;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-19 20:06:29 +12:00
|
|
|
int taskLoad() override { return 100; }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-19 20:06:29 +12:00
|
|
|
void onFinished(TThread::RunnableP) override;
|
2016-03-19 06:57:51 +13:00
|
|
|
};
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
// Implementations
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//==================
|
|
|
|
// TRenderer
|
|
|
|
//------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderer::TRenderer(int nThread) {
|
|
|
|
m_imp = new TRendererImp(nThread); // Already adds a ref
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderer::TRenderer(TRendererImp *imp) : m_imp(imp) {
|
|
|
|
if (m_imp) m_imp->addRef();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderer::TRenderer(const TRenderer &r) : m_imp(r.m_imp) {
|
|
|
|
if (m_imp) m_imp->addRef();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::operator=(const TRenderer &r) {
|
|
|
|
m_imp = r.m_imp;
|
|
|
|
if (m_imp) m_imp->addRef();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderer::~TRenderer() {
|
|
|
|
if (m_imp) m_imp->release();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! The static method is intended for use inside a rendering thread only. It
|
|
|
|
//! returns
|
2016-03-19 06:57:51 +13:00
|
|
|
//! a copy of the eventual TRenderer interface installed on the thread -
|
|
|
|
//! or an empty TRenderer if none happened to be. It can be set using the
|
|
|
|
//! install() and uninstall() methods.
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderer TRenderer::instance() {
|
|
|
|
TRendererImp **lData = rendererStorage.localData();
|
|
|
|
if (lData) return TRenderer(*lData);
|
|
|
|
return 0;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Returns the renderer id.
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRenderer::rendererId() { return m_imp->m_rendererId; }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Returns the rendering process id currently running on the invoking
|
|
|
|
//! thread.
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRenderer::renderId() {
|
|
|
|
unsigned long *lData = renderIdsStorage.localData();
|
|
|
|
return lData ? *lData : -1;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Builds and returns an id for starting a new rendering process.
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRenderer::buildRenderId() {
|
|
|
|
return TRendererImp::m_renderIdCounter++;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Returns the renderId that will be used by the next startRendering() call.
|
|
|
|
//! This method can be used to retrieve the renderId of a rendering instance
|
|
|
|
//! before it is actually started - provided that no other render instance
|
|
|
|
//! is launched inbetween.
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRenderer::nextRenderId() {
|
|
|
|
return TRendererImp::m_renderIdCounter;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void TRendererImp::declareRenderStart(unsigned long renderId) {
|
|
|
|
// Inform the resource managers
|
|
|
|
for (unsigned int i = 0; i < m_managers.size(); ++i)
|
|
|
|
m_managers[i]->onRenderInstanceStart(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//! Declares that the render process of passed renderId is about to start.
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::declareRenderStart(unsigned long renderId) {
|
|
|
|
m_imp->declareRenderStart(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void TRendererImp::declareRenderEnd(unsigned long renderId) {
|
|
|
|
// Inform the resource managers
|
|
|
|
for (int i = m_managers.size() - 1; i >= 0; --i)
|
|
|
|
m_managers[i]->onRenderInstanceEnd(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//! Declares that the render process of passed renderId has ended.
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::declareRenderEnd(unsigned long renderId) {
|
|
|
|
m_imp->declareRenderEnd(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void TRendererImp::declareFrameStart(double frame) {
|
|
|
|
// Inform the resource managers
|
|
|
|
for (unsigned int i = 0; i < m_managers.size(); ++i)
|
|
|
|
m_managers[i]->onRenderFrameStart(frame);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//! Declares that render of passed frame is about to start.
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::declareFrameStart(double frame) {
|
|
|
|
m_imp->declareFrameStart(frame);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void TRendererImp::declareFrameEnd(double frame) {
|
|
|
|
// Inform the resource managers
|
|
|
|
for (int i = m_managers.size() - 1; i >= 0; --i)
|
|
|
|
m_managers[i]->onRenderFrameEnd(frame);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//! Declares that render of passed has ended.
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::declareFrameEnd(double frame) { m_imp->declareFrameEnd(frame); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void TRendererImp::declareStatusStart(int renderStatus) {
|
|
|
|
// Inform the resource managers
|
|
|
|
for (unsigned int i = 0; i < m_managers.size(); ++i)
|
|
|
|
m_managers[i]->onRenderStatusStart(renderStatus);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
inline void TRendererImp::declareStatusEnd(int renderStatus) {
|
|
|
|
// Inform the resource managers
|
|
|
|
for (int i = m_managers.size() - 1; i >= 0; --i)
|
|
|
|
m_managers[i]->onRenderStatusEnd(renderStatus);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Installs the specified render process on the invoking thread.
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::install(unsigned long renderId) {
|
|
|
|
m_imp->addRef();
|
|
|
|
rendererStorage.setLocalData(new (TRendererImp *)(m_imp));
|
|
|
|
renderIdsStorage.setLocalData(new unsigned long(renderId));
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Uninstalls any rendering process active on the invoking thread.
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::uninstall() {
|
|
|
|
rendererStorage.setLocalData(0);
|
|
|
|
renderIdsStorage.setLocalData(0);
|
|
|
|
m_imp->release();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderResourceManager *TRenderer::getManager(unsigned int id) const {
|
|
|
|
return m_imp->m_managers[id];
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::enablePrecomputing(bool on) { m_imp->enablePrecomputing(on); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool TRenderer::isPrecomputingEnabled() const {
|
|
|
|
return m_imp->isPrecomputingEnabled();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::setThreadsCount(int nThreads) {
|
|
|
|
m_imp->setThreadsCount(nThreads);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::addPort(TRenderPort *port) { m_imp->addPort(port); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::removePort(TRenderPort *port) { m_imp->removePort(port); }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
2020-10-08 10:27:55 +13:00
|
|
|
// Looks like this only handles a single frame?
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRenderer::startRendering(double f, const TRenderSettings &info,
|
|
|
|
const TFxPair &actualRoot) {
|
|
|
|
assert(f >= 0);
|
|
|
|
|
|
|
|
std::vector<RenderData> *rds = new std::vector<RenderData>;
|
|
|
|
rds->push_back(RenderData(f, info, actualRoot));
|
|
|
|
return startRendering(rds);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
|
|
|
//! Queues a rendering event in the main event loop.
|
2016-06-15 18:43:10 +12:00
|
|
|
//! NOTE: The passed pointer is owned by the TRenderer after it is submitted for
|
|
|
|
//! rendering -
|
2016-03-19 06:57:51 +13:00
|
|
|
//! do not delete it later.
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned long TRenderer::startRendering(
|
|
|
|
const std::vector<RenderData> *renderDatas) {
|
|
|
|
if (renderDatas->empty()) {
|
|
|
|
delete renderDatas;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build a new render Id
|
|
|
|
unsigned long renderId = m_imp->m_renderIdCounter++;
|
|
|
|
|
|
|
|
TRendererStartInvoker::StartInvokerRenderData srd;
|
|
|
|
srd.m_renderId = renderId;
|
|
|
|
srd.m_renderDataVector = renderDatas;
|
|
|
|
TRendererStartInvoker::instance()->emitStartRender(m_imp, srd);
|
|
|
|
|
|
|
|
return renderId;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::abortRendering(unsigned long renderId) {
|
|
|
|
m_imp->abortRendering(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRenderer::stopRendering(bool waitForCompleteStop) {
|
|
|
|
m_imp->stopRendering(waitForCompleteStop);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool TRenderer::isAborted(unsigned long renderId) const {
|
|
|
|
return m_imp->hasToDie(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
int TRenderer::getRenderStatus(unsigned long renderId) const {
|
|
|
|
return m_imp->getRenderStatus(renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//=====================
|
|
|
|
// TRendererImp
|
|
|
|
//---------------------
|
|
|
|
|
|
|
|
TRendererImp::TRendererImp(int nThreads)
|
2016-06-15 18:43:10 +12:00
|
|
|
: m_executor()
|
|
|
|
, m_undoneTasks()
|
|
|
|
, m_rendererId(m_rendererIdCounter++)
|
|
|
|
, m_precomputingEnabled(true) {
|
|
|
|
m_executor.setMaxActiveTasks(nThreads);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
std::vector<TRenderResourceManagerGenerator *> &generators =
|
|
|
|
TRenderResourceManagerGenerator::generators(false);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// May be adopted by other TRenderers from now on.
|
|
|
|
addRef();
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
rendererStorage.setLocalData(new (TRendererImp *)(this));
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < generators.size(); ++i) {
|
|
|
|
TRenderResourceManager *manager = (*generators[i])();
|
|
|
|
if (manager) m_managers.push_back(manager);
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
rendererStorage.setLocalData(0);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRendererImp::~TRendererImp() {
|
|
|
|
rendererStorage.setLocalData(new (TRendererImp *)(this));
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
int i;
|
|
|
|
for (i = m_managers.size() - 1; i >= 0; --i)
|
|
|
|
if (m_managers[i]->renderHasOwnership()) delete m_managers[i];
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
rendererStorage.setLocalData(0);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::addPort(TRenderPort *port) {
|
|
|
|
QWriteLocker sl(&m_portsLock);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
PortContainerIterator it = std::find(m_ports.begin(), m_ports.end(), port);
|
|
|
|
if (it == m_ports.end()) m_ports.push_back(port);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::removePort(TRenderPort *port) {
|
|
|
|
QWriteLocker sl(&m_portsLock);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
PortContainerIterator it = std::find(m_ports.begin(), m_ports.end(), port);
|
|
|
|
if (it != m_ports.end()) m_ports.erase(it);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool TRendererImp::hasToDie(unsigned long renderId) {
|
|
|
|
QMutexLocker sl(&m_renderInstancesMutex);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
std::map<unsigned long, RenderInstanceInfos>::iterator it =
|
|
|
|
m_activeInstances.find(renderId);
|
|
|
|
assert(it != m_activeInstances.end());
|
|
|
|
return it == m_activeInstances.end() ? true : it->second.m_canceled;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
int TRendererImp::getRenderStatus(unsigned long renderId) {
|
|
|
|
QMutexLocker sl(&m_renderInstancesMutex);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
std::map<unsigned long, RenderInstanceInfos>::iterator it =
|
|
|
|
m_activeInstances.find(renderId);
|
|
|
|
assert(it != m_activeInstances.end());
|
|
|
|
return it == m_activeInstances.end() ? true : it->second.m_status;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::abortRendering(unsigned long renderId) {
|
|
|
|
QMutexLocker sl(&m_renderInstancesMutex);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
std::map<unsigned long, RenderInstanceInfos>::iterator it =
|
|
|
|
m_activeInstances.find(renderId);
|
|
|
|
if (it != m_activeInstances.end()) it->second.m_canceled = true;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::stopRendering(bool waitForCompleteStop) {
|
|
|
|
QMutexLocker sl(&m_renderInstancesMutex);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
{
|
|
|
|
// Tasks already stop rendering on their own when they don't find their
|
|
|
|
// render ids here.
|
|
|
|
std::map<unsigned long, RenderInstanceInfos>::iterator it;
|
|
|
|
for (it = m_activeInstances.begin(); it != m_activeInstances.end(); ++it)
|
|
|
|
it->second.m_canceled = true;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
if (waitForCompleteStop && m_undoneTasks > 0) {
|
|
|
|
// Sometimes, QEventLoop suddenly stops processing slots (especially those
|
|
|
|
// notifications
|
|
|
|
// from active rendering instances) - therefore resulting in a block of the
|
|
|
|
// application.
|
|
|
|
// I've not figured out why (#QTBUG-11649?) - but substituting with a plain
|
|
|
|
// while
|
|
|
|
// seems to do the trick...
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
/*QEventLoop eventLoop;
|
|
|
|
m_waitingLoops.push_back(&eventLoop);
|
|
|
|
eventLoop.exec();*/
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
bool loopQuit = false;
|
|
|
|
m_waitingLoops.push_back(&loopQuit);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
sl.unlock();
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
while (!loopQuit)
|
|
|
|
QCoreApplication::processEvents(QEventLoop::AllEvents |
|
|
|
|
QEventLoop::WaitForMoreEvents);
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::quitWaitingLoops() {
|
|
|
|
// Make the stopRendering waiting loops quit
|
|
|
|
while (!m_waitingLoops.empty()) {
|
|
|
|
// rendererImp->m_waitingLoops.back()->quit();
|
|
|
|
*m_waitingLoops.back() = true;
|
|
|
|
m_waitingLoops.pop_back();
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::notifyRasterStarted(const TRenderPort::RenderData &rd) {
|
|
|
|
// Since notifications may trigger port removals, we always work on a copy of
|
|
|
|
// the ports
|
|
|
|
// vector.
|
|
|
|
TRendererImp::PortContainer portsCopy;
|
|
|
|
{
|
|
|
|
QReadLocker sl(&m_portsLock);
|
|
|
|
portsCopy = m_ports;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PortContainerIterator it = portsCopy.begin(); it != portsCopy.end();
|
|
|
|
++it)
|
|
|
|
(*it)->onRenderRasterStarted(rd);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::notifyRasterCompleted(const TRenderPort::RenderData &rd) {
|
|
|
|
TRendererImp::PortContainer portsCopy;
|
|
|
|
{
|
|
|
|
QReadLocker sl(&m_portsLock);
|
|
|
|
portsCopy = m_ports;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
assert(rd.m_rasA);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
for (PortContainerIterator it = portsCopy.begin(); it != portsCopy.end();
|
|
|
|
++it)
|
|
|
|
(*it)->onRenderRasterCompleted(rd);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::notifyRasterFailure(const TRenderPort::RenderData &rd,
|
|
|
|
TException &e) {
|
|
|
|
TRendererImp::PortContainer portsCopy;
|
|
|
|
{
|
|
|
|
QReadLocker sl(&m_portsLock);
|
|
|
|
portsCopy = m_ports;
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
for (PortContainerIterator it = portsCopy.begin(); it != portsCopy.end();
|
|
|
|
++it)
|
|
|
|
(*it)->onRenderFailure(rd, e);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::notifyRenderFinished(bool isCanceled) {
|
|
|
|
TRendererImp::PortContainer portsCopy;
|
|
|
|
{
|
|
|
|
QReadLocker sl(&m_portsLock);
|
|
|
|
portsCopy = m_ports;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto sortedFxs = calculateSortedFxs(rootFx);
|
|
|
|
for (auto fx : sortedFxs) {
|
|
|
|
if (fx) const_cast<TFx *>(fx)->callEndRenderHandler();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PortContainerIterator it = portsCopy.begin(); it != portsCopy.end();
|
|
|
|
++it)
|
|
|
|
(*it)->onRenderFinished();
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//====================
|
|
|
|
// TRenderPort
|
|
|
|
//--------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderPort::TRenderPort() {}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderPort::~TRenderPort() {}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! Setta \b m_renderArea a \b area e pulisce l'istanza corrente di \b
|
|
|
|
//! RasterPool.
|
|
|
|
void TRenderPort::setRenderArea(const TRectD &area) { m_renderArea = area; }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//! Ritorna \b m_renderArea.
|
|
|
|
TRectD &TRenderPort::getRenderArea() { return m_renderArea; }
|
2016-03-19 06:57:51 +13:00
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
|
|
|
|
//===================
|
|
|
|
// RenderTask
|
|
|
|
//-------------------
|
|
|
|
|
|
|
|
RenderTask::RenderTask(unsigned long renderId, unsigned long taskId,
|
2016-06-15 18:43:10 +12:00
|
|
|
double frame, const TRenderSettings &ri,
|
|
|
|
const TFxPair &fx, const TPointD &framePos,
|
|
|
|
const TDimension &frameSize,
|
|
|
|
const TRendererImpP &rendererImp)
|
|
|
|
: m_renderId(renderId)
|
|
|
|
, m_taskId(taskId)
|
|
|
|
, m_info(ri)
|
|
|
|
, m_fx(fx)
|
|
|
|
, m_frameSize(frameSize)
|
|
|
|
, m_framePos(framePos)
|
|
|
|
, m_rendererImp(rendererImp)
|
|
|
|
, m_fieldRender(ri.m_fieldPrevalence != TRenderSettings::NoField)
|
|
|
|
, m_stereoscopic(ri.m_stereoscopic) {
|
|
|
|
m_frames.push_back(frame);
|
|
|
|
|
|
|
|
// Connect the onFinished slot
|
|
|
|
connect(this, SIGNAL(finished(TThread::RunnableP)), this,
|
|
|
|
SLOT(onFinished(TThread::RunnableP)));
|
|
|
|
connect(this, SIGNAL(exception(TThread::RunnableP)), this,
|
|
|
|
SLOT(onFinished(TThread::RunnableP)));
|
|
|
|
|
|
|
|
// The shrink info is currently reversed to the settings'affine. Shrink info
|
|
|
|
// in the TRenderSettings
|
|
|
|
// is no longer supported.
|
|
|
|
m_info.m_shrinkX = m_info.m_shrinkY = 1;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::preRun() {
|
|
|
|
TRectD geom(m_framePos, TDimensionD(m_frameSize.lx, m_frameSize.ly));
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
if (m_fx.m_frameA) m_fx.m_frameA->dryCompute(geom, m_frames[0], m_info);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
if (m_fx.m_frameB)
|
|
|
|
m_fx.m_frameB->dryCompute(
|
|
|
|
geom, m_fieldRender ? m_frames[0] + 0.5 : m_frames[0], m_info);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::run() {
|
|
|
|
// Retrieve the task's frame
|
|
|
|
assert(!m_frames.empty());
|
|
|
|
double t = m_frames[0];
|
|
|
|
|
|
|
|
if (m_rendererImp->hasToDie(m_renderId)) {
|
|
|
|
TException e("Render task aborted");
|
|
|
|
onFrameFailed(e);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Install the renderer in current thread
|
|
|
|
rendererStorage.setLocalData(
|
|
|
|
new (TRendererImp *)(m_rendererImp.getPointer()));
|
|
|
|
renderIdsStorage.setLocalData(new unsigned long(m_renderId));
|
|
|
|
|
|
|
|
// Inform the managers of frame start
|
|
|
|
m_rendererImp->declareFrameStart(t);
|
|
|
|
|
|
|
|
auto sortedFxs = calculateSortedFxs(m_fx.m_frameA);
|
|
|
|
for (auto fx : sortedFxs) {
|
|
|
|
if (fx) const_cast<TFx *>(fx)->callStartRenderFrameHandler(&m_info, t);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
onFrameStarted();
|
|
|
|
|
|
|
|
TStopWatch::global(8).start();
|
|
|
|
|
|
|
|
if (!m_fieldRender && !m_stereoscopic) {
|
|
|
|
// Common case - just build the first tile
|
|
|
|
buildTile(m_tileA);
|
2020-10-08 10:27:55 +13:00
|
|
|
// static int iCount = 0;
|
|
|
|
// QString qPath("C:\\butta\\image_" +
|
|
|
|
// QString::number(++iCount).rightJustified(3, '0') + ".tif");
|
|
|
|
// TImageWriter::save(TFilePath(qPath.toStdWString()), m_tileA.getRaster());
|
|
|
|
/*-- Normally this is the Fx rendering process --*/
|
2016-06-15 18:43:10 +12:00
|
|
|
m_fx.m_frameA->compute(m_tileA, t, m_info);
|
2020-10-08 10:27:55 +13:00
|
|
|
// The tile now has the image.
|
|
|
|
// TImageWriter::save(TFilePath(qPath.toStdWString()), m_tileA.getRaster());
|
2016-06-15 18:43:10 +12:00
|
|
|
} else {
|
|
|
|
assert(!(m_stereoscopic && m_fieldRender));
|
|
|
|
// Field rendering or stereoscopic case
|
|
|
|
if (m_stereoscopic) {
|
|
|
|
buildTile(m_tileA);
|
|
|
|
m_fx.m_frameA->compute(m_tileA, t, m_info);
|
|
|
|
|
|
|
|
buildTile(m_tileB);
|
|
|
|
m_fx.m_frameB->compute(m_tileB, t, m_info);
|
|
|
|
}
|
|
|
|
// if fieldPrevalence, Decide the rendering frames depending on field
|
|
|
|
// prevalence
|
|
|
|
else if (m_info.m_fieldPrevalence == TRenderSettings::EvenField) {
|
|
|
|
buildTile(m_tileA);
|
|
|
|
m_fx.m_frameA->compute(m_tileA, t, m_info);
|
|
|
|
|
|
|
|
buildTile(m_tileB);
|
|
|
|
m_fx.m_frameB->compute(m_tileB, t + 0.5, m_info);
|
|
|
|
} else {
|
|
|
|
buildTile(m_tileB);
|
|
|
|
m_fx.m_frameA->compute(m_tileB, t, m_info);
|
|
|
|
|
|
|
|
buildTile(m_tileA);
|
|
|
|
m_fx.m_frameB->compute(m_tileA, t + 0.5, m_info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TStopWatch::global(8).stop();
|
|
|
|
|
|
|
|
onFrameCompleted();
|
|
|
|
} catch (TException &e) {
|
|
|
|
onFrameFailed(e);
|
|
|
|
} catch (...) {
|
|
|
|
TException ex("Unknown render exception");
|
|
|
|
onFrameFailed(ex);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Inform the managers of frame end
|
|
|
|
m_rendererImp->declareFrameEnd(t);
|
|
|
|
|
|
|
|
// Uninstall the renderer from current thread
|
|
|
|
rendererStorage.setLocalData(0);
|
|
|
|
renderIdsStorage.setLocalData(0);
|
|
|
|
|
|
|
|
for (auto fx : sortedFxs) {
|
|
|
|
if (fx) const_cast<TFx *>(fx)->callEndRenderFrameHandler(&m_info, t);
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::buildTile(TTile &tile) {
|
|
|
|
tile.m_pos = m_framePos;
|
|
|
|
tile.setRaster(
|
|
|
|
m_rendererImp->m_rasterPool.getRaster(m_frameSize, m_info.m_bpp));
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::releaseTiles() {
|
|
|
|
m_rendererImp->m_rasterPool.releaseRaster(m_tileA.getRaster());
|
|
|
|
m_tileA.setRaster(TRasterP());
|
|
|
|
if (m_fieldRender || m_stereoscopic) {
|
|
|
|
m_rendererImp->m_rasterPool.releaseRaster(m_tileB.getRaster());
|
|
|
|
m_tileB.setRaster(TRasterP());
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::onFrameStarted() {
|
|
|
|
TRenderPort::RenderData rd(m_frames, m_info, 0, 0, m_renderId, m_taskId);
|
|
|
|
m_rendererImp->notifyRasterStarted(rd);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::onFrameCompleted() {
|
|
|
|
TRasterP rasA(m_tileA.getRaster());
|
|
|
|
TRasterP rasB(m_tileB.getRaster());
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
if (m_fieldRender) {
|
|
|
|
assert(rasB);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
double t = m_frames[0];
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
int f = (m_info.m_fieldPrevalence == TRenderSettings::EvenField) ? 0 : 1;
|
|
|
|
interlace(rasA, rasB, f);
|
|
|
|
rasB = TRasterP();
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderPort::RenderData rd(m_frames, m_info, rasA, rasB, m_renderId,
|
|
|
|
m_taskId);
|
|
|
|
m_rendererImp->notifyRasterCompleted(rd);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::onFrameFailed(TException &e) {
|
|
|
|
// TRasterP evenRas(m_evenTile.getRaster());
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderPort::RenderData rd(m_frames, m_info, m_tileA.getRaster(),
|
|
|
|
m_tileB.getRaster(), m_renderId, m_taskId);
|
|
|
|
m_rendererImp->notifyRasterFailure(rd, e);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void RenderTask::onFinished(TThread::RunnableP) {
|
|
|
|
TRendererImp *rendererImp = m_rendererImp.getPointer();
|
|
|
|
--rendererImp->m_undoneTasks;
|
|
|
|
|
|
|
|
// Tiles release back to the Raster Pool happens in the main thread, after all
|
|
|
|
// possible
|
|
|
|
// signals emitted in the onFrameCompleted/Failed notifications have been
|
|
|
|
// resolved, thus
|
|
|
|
// ensuring that no other rendering thread owns the rasters before them.
|
|
|
|
releaseTiles();
|
|
|
|
|
|
|
|
// Update the render instance status
|
|
|
|
bool instanceExpires = false;
|
2016-08-04 19:23:36 +12:00
|
|
|
bool isCanceled = false;
|
2016-06-15 18:43:10 +12:00
|
|
|
{
|
|
|
|
QMutexLocker sl(&rendererImp->m_renderInstancesMutex);
|
|
|
|
std::map<unsigned long, TRendererImp::RenderInstanceInfos>::iterator it =
|
|
|
|
rendererImp->m_activeInstances.find(m_renderId);
|
|
|
|
|
|
|
|
if (it != rendererImp->m_activeInstances.end() &&
|
|
|
|
(--it->second.m_activeTasks) <= 0) {
|
|
|
|
instanceExpires = true;
|
2016-08-04 19:23:36 +12:00
|
|
|
isCanceled = (m_info.m_isCanceled && *m_info.m_isCanceled);
|
2016-06-15 18:43:10 +12:00
|
|
|
rendererImp->m_activeInstances.erase(m_renderId);
|
2016-07-13 21:55:14 +12:00
|
|
|
// m_info is freed, don't access further!
|
2016-06-15 18:43:10 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the render instance has just expired
|
|
|
|
if (instanceExpires) {
|
|
|
|
/*-- キャンセルされた場合はm_overallRenderedRegionの更新をしない --*/
|
|
|
|
|
|
|
|
// Inform the render ports
|
|
|
|
rendererImp->notifyRenderFinished(isCanceled);
|
|
|
|
|
|
|
|
// NOTE: This slot is currently invoked on the main thread. It could
|
|
|
|
// eventually be
|
|
|
|
// invoked directly on rendering threads, specifying the
|
|
|
|
// Qt::DirectConnection option -
|
|
|
|
// but probably there would be no real advantage in doing so...
|
|
|
|
|
|
|
|
// Temporarily install the renderer in current thread
|
|
|
|
rendererStorage.setLocalData(new (TRendererImp *)(rendererImp));
|
|
|
|
renderIdsStorage.setLocalData(new unsigned long(m_renderId));
|
|
|
|
|
|
|
|
// Inform the resource managers
|
|
|
|
rendererImp->declareRenderEnd(m_renderId);
|
|
|
|
|
|
|
|
// Uninstall the temporary
|
|
|
|
rendererStorage.setLocalData(0);
|
|
|
|
renderIdsStorage.setLocalData(0);
|
|
|
|
|
|
|
|
rendererImp->m_rasterPool
|
|
|
|
.clear(); // Isn't this misplaced? Should be in the block
|
|
|
|
} // below...
|
|
|
|
|
|
|
|
// If no rendering task (of this or other render instances) is found...
|
|
|
|
if (rendererImp->m_undoneTasks == 0) {
|
|
|
|
QMutexLocker sl(&rendererImp->m_renderInstancesMutex);
|
|
|
|
rendererImp->quitWaitingLoops();
|
|
|
|
}
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//================================================================================
|
|
|
|
// Tough Stuff
|
|
|
|
//================================================================================
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererStartInvoker::emitStartRender(TRendererImp *renderer,
|
|
|
|
StartInvokerRenderData rd) {
|
|
|
|
renderer->addRef();
|
|
|
|
Q_EMIT startRender(renderer, rd);
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererStartInvoker::doStartRender(TRendererImp *renderer,
|
|
|
|
StartInvokerRenderData rd) {
|
|
|
|
renderer->startRendering(rd.m_renderId, *rd.m_renderDataVector);
|
|
|
|
renderer->release();
|
|
|
|
delete rd.m_renderDataVector;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<const TFx *> calculateSortedFxs(TRasterFxP rootFx) {
|
2020-10-08 10:27:55 +13:00
|
|
|
std::map<const TFx *, std::set<const TFx *>> E;
|
|
|
|
std::set<const TFx *> Sources;
|
2016-06-15 18:43:10 +12:00
|
|
|
|
|
|
|
std::queue<const TFx *> Q;
|
|
|
|
Q.push(rootFx.getPointer());
|
|
|
|
|
|
|
|
E[rootFx.getPointer()] = std::set<const TFx *>();
|
|
|
|
|
|
|
|
while (!Q.empty()) {
|
|
|
|
const TFx *vptr = Q.front();
|
|
|
|
Q.pop();
|
|
|
|
if (!vptr) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-10-08 10:27:55 +13:00
|
|
|
/* Visit the Fx beyond the connected input port
|
|
|
|
Exit if there is no input port */
|
2016-06-15 18:43:10 +12:00
|
|
|
int portCount = vptr->getInputPortCount();
|
|
|
|
if (portCount < 1) {
|
|
|
|
Sources.insert(vptr);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < portCount; i++) {
|
|
|
|
TFxPort *port = vptr->getInputPort(i);
|
|
|
|
if (!port) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
TFxP u = port->getFx();
|
|
|
|
const TFx *uptr = u.getPointer();
|
|
|
|
if (E.count(uptr) == 0) {
|
|
|
|
E[uptr] = std::set<const TFx *>();
|
|
|
|
}
|
|
|
|
if (E[uptr].count(vptr) == 0) {
|
|
|
|
E[uptr].insert(vptr);
|
|
|
|
}
|
|
|
|
Q.push(uptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-08 10:27:55 +13:00
|
|
|
/* Topological sort */
|
2016-06-15 18:43:10 +12:00
|
|
|
std::set<const TFx *> visited;
|
|
|
|
std::vector<const TFx *> L;
|
|
|
|
std::function<void(const TFx *)> visit = [&visit, &visited, &E,
|
|
|
|
&L](const TFx *fx) {
|
|
|
|
if (visited.count(fx)) return;
|
|
|
|
visited.insert(fx);
|
|
|
|
auto edge = E[fx];
|
|
|
|
for (auto i = edge.cbegin(); i != edge.cend(); i++) {
|
|
|
|
visit(*i);
|
|
|
|
}
|
|
|
|
L.insert(L.begin(), fx);
|
|
|
|
};
|
|
|
|
for (auto i = E.cbegin(); i != E.cend(); i++) {
|
|
|
|
visit(i->first);
|
|
|
|
}
|
|
|
|
return L;
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
//---------------------------------------------------------
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
void TRendererImp::startRendering(
|
|
|
|
unsigned long renderId,
|
|
|
|
const std::vector<TRenderer::RenderData> &renderDatas) {
|
|
|
|
rootFx = renderDatas.front().m_fxRoot.m_frameA;
|
|
|
|
int T = renderDatas.size();
|
|
|
|
for (int z = 0; z < T; z++) {
|
|
|
|
auto sortedFxs = calculateSortedFxs(renderDatas[z].m_fxRoot.m_frameA);
|
|
|
|
if (z == 0) {
|
|
|
|
for (auto fx : sortedFxs) {
|
|
|
|
if (fx) const_cast<TFx *>(fx)->callStartRenderHandler();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct locals {
|
|
|
|
static inline void setStorage(TRendererImp *imp, unsigned long renderId) {
|
|
|
|
rendererStorage.setLocalData(new (TRendererImp *)(imp));
|
|
|
|
renderIdsStorage.setLocalData(new unsigned long(renderId));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void clearStorage() {
|
|
|
|
rendererStorage.setLocalData(0);
|
|
|
|
renderIdsStorage.setLocalData(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void declareStatusStart(TRendererImp *imp,
|
|
|
|
TRenderer::RenderStatus status,
|
|
|
|
RenderInstanceInfos *renderInfos) {
|
|
|
|
renderInfos->m_status = status;
|
|
|
|
imp->declareStatusStart(status);
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
|
|
|
|
|
|
struct InstanceDeclaration {
|
|
|
|
TRendererImp *m_imp;
|
|
|
|
unsigned long m_renderId;
|
|
|
|
bool m_rollback;
|
|
|
|
|
|
|
|
InstanceDeclaration(TRendererImp *imp, unsigned long renderId)
|
|
|
|
: m_imp(imp), m_renderId(renderId), m_rollback(true) {}
|
|
|
|
|
|
|
|
~InstanceDeclaration() {
|
|
|
|
if (m_rollback) {
|
|
|
|
QMutexLocker locker(&m_imp->m_renderInstancesMutex);
|
|
|
|
|
|
|
|
m_imp->m_activeInstances.erase(m_renderId);
|
|
|
|
if (m_imp->m_undoneTasks == 0) m_imp->quitWaitingLoops();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void commit() { m_rollback = false; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct StorageDeclaration {
|
|
|
|
StorageDeclaration(TRendererImp *imp, unsigned long renderId) {
|
|
|
|
setStorage(imp, renderId);
|
|
|
|
}
|
|
|
|
|
|
|
|
~StorageDeclaration() { clearStorage(); }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct RenderDeclaration {
|
|
|
|
TRendererImp *m_imp;
|
|
|
|
unsigned long m_renderId;
|
|
|
|
bool m_rollback;
|
|
|
|
|
|
|
|
RenderDeclaration(TRendererImp *imp, unsigned long renderId)
|
|
|
|
: m_imp(imp), m_renderId(renderId), m_rollback(true) {
|
|
|
|
imp->declareRenderStart(renderId);
|
|
|
|
}
|
|
|
|
|
|
|
|
~RenderDeclaration() {
|
|
|
|
if (m_rollback) m_imp->declareRenderEnd(m_renderId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void commit() { m_rollback = false; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct StatusDeclaration {
|
|
|
|
TRendererImp *m_imp;
|
|
|
|
TRenderer::RenderStatus m_status;
|
|
|
|
|
|
|
|
StatusDeclaration(TRendererImp *imp, TRenderer::RenderStatus status,
|
|
|
|
RenderInstanceInfos *renderInfos)
|
|
|
|
: m_imp(imp), m_status(status) {
|
|
|
|
declareStatusStart(imp, status, renderInfos);
|
|
|
|
}
|
|
|
|
|
|
|
|
~StatusDeclaration() { m_imp->declareStatusEnd(m_status); }
|
|
|
|
};
|
|
|
|
}; // locals
|
|
|
|
|
|
|
|
// DIAGNOSTICS_CLEAR;
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Preliminary initializations
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
// Calculate the overall render area - sum of all render ports' areas
|
|
|
|
TRectD renderArea;
|
|
|
|
{
|
|
|
|
QReadLocker sl(&m_portsLock);
|
|
|
|
|
|
|
|
for (PortContainerIterator it = m_ports.begin(); it != m_ports.end(); ++it)
|
|
|
|
renderArea += (*it)->getRenderArea();
|
|
|
|
}
|
|
|
|
|
|
|
|
const TRenderSettings &info(renderDatas[0].m_info);
|
|
|
|
|
|
|
|
// Extract the render geometry
|
|
|
|
TPointD pos(renderArea.getP00());
|
|
|
|
TDimension frameSize(tceil(renderArea.getLx()), tceil(renderArea.getLy()));
|
|
|
|
|
|
|
|
TRectD camBox(TPointD(pos.x / info.m_shrinkX, pos.y / info.m_shrinkY),
|
|
|
|
TDimensionD(frameSize.lx, frameSize.ly));
|
|
|
|
|
|
|
|
// Refresh the raster pool specs
|
|
|
|
m_rasterPool.setRasterSpecs(frameSize, info.m_bpp);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// Set a temporary active instance count - so that hasToDie(renderId) returns
|
|
|
|
// false
|
|
|
|
RenderInstanceInfos *renderInfos;
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_renderInstancesMutex);
|
|
|
|
renderInfos = &m_activeInstances[renderId];
|
|
|
|
renderInfos->m_activeTasks = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
locals::InstanceDeclaration instanceDecl(this, renderId);
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Clustering - Render Tasks creation
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
std::vector<RenderTask *> tasksVector;
|
|
|
|
|
|
|
|
struct TasksCleaner {
|
|
|
|
std::vector<RenderTask *> &m_tasksVector;
|
|
|
|
~TasksCleaner() {
|
|
|
|
std::for_each(m_tasksVector.begin(), m_tasksVector.end(),
|
2019-02-21 15:58:39 +13:00
|
|
|
std::default_delete<RenderTask>());
|
2016-06-15 18:43:10 +12:00
|
|
|
}
|
|
|
|
} tasksCleaner = {tasksVector};
|
|
|
|
|
|
|
|
unsigned long tasksIdCounter = 0;
|
|
|
|
|
|
|
|
std::map<std::string, RenderTask *> clusters;
|
|
|
|
std::vector<TRenderer::RenderData>::const_iterator it;
|
|
|
|
std::map<std::string, RenderTask *>::iterator jt;
|
|
|
|
|
|
|
|
for (it = renderDatas.begin(); it != renderDatas.end(); ++it) {
|
|
|
|
// Check for user cancels
|
|
|
|
if (hasToDie(renderId)) return;
|
|
|
|
|
|
|
|
// Build the frame's description alias
|
|
|
|
const TRenderer::RenderData &renderData = *it;
|
|
|
|
|
2020-10-08 10:27:55 +13:00
|
|
|
/*--- Camera size (used for Level Auto and noise) ---*/
|
2016-06-15 18:43:10 +12:00
|
|
|
TRenderSettings rs = renderData.m_info;
|
|
|
|
rs.m_cameraBox = camBox;
|
2020-10-08 10:27:55 +13:00
|
|
|
/*--- Flag when Preview calculation is canceled in the middle ---*/
|
2016-06-15 18:43:10 +12:00
|
|
|
rs.m_isCanceled = &renderInfos->m_canceled;
|
|
|
|
|
|
|
|
TRasterFxP fx = renderData.m_fxRoot.m_frameA;
|
|
|
|
assert(fx);
|
|
|
|
|
|
|
|
double frame = renderData.m_frame;
|
|
|
|
|
|
|
|
std::string alias = fx->getAlias(frame, renderData.m_info);
|
|
|
|
if (renderData.m_fxRoot.m_frameB)
|
|
|
|
alias = alias +
|
|
|
|
renderData.m_fxRoot.m_frameB->getAlias(frame, renderData.m_info);
|
|
|
|
|
2018-01-12 20:00:36 +13:00
|
|
|
// If the render contains offscreen render, then prepare the
|
|
|
|
// QOffscreenSurface
|
|
|
|
// in main (GUI) thread. For now it is used only in the plasticDeformerFx.
|
|
|
|
if (alias.find("plasticDeformerFx") != std::string::npos &&
|
|
|
|
QThread::currentThread() == qGuiApp->thread()) {
|
|
|
|
rs.m_offScreenSurface.reset(new QOffscreenSurface());
|
|
|
|
rs.m_offScreenSurface->setFormat(QSurfaceFormat::defaultFormat());
|
|
|
|
rs.m_offScreenSurface->create();
|
|
|
|
}
|
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
// Search the alias among stored clusters - and store the frame
|
|
|
|
jt = clusters.find(alias);
|
|
|
|
|
|
|
|
if (jt == clusters.end()) {
|
|
|
|
RenderTask *newTask =
|
|
|
|
new RenderTask(renderId, tasksIdCounter++, renderData.m_frame, rs,
|
|
|
|
renderData.m_fxRoot, pos, frameSize, this);
|
|
|
|
|
|
|
|
tasksVector.push_back(newTask);
|
|
|
|
clusters.insert(std::make_pair(alias, newTask));
|
|
|
|
} else
|
|
|
|
jt->second->addFrame(renderData.m_frame);
|
|
|
|
|
|
|
|
// Call processEvents to make the GUI reactive.
|
|
|
|
QCoreApplication::instance()->processEvents();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release the clusters - we'll just need the tasks vector from now on
|
|
|
|
clusters.clear();
|
|
|
|
|
|
|
|
std::vector<RenderTask *>::iterator kt, kEnd = tasksVector.end();
|
|
|
|
{
|
|
|
|
// Install TRenderer on current thread before proceeding
|
|
|
|
locals::StorageDeclaration storageDecl(this, renderId);
|
|
|
|
|
|
|
|
// Inform the resource managers
|
|
|
|
locals::RenderDeclaration renderDecl(this, renderId);
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Precomputing
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
if (m_precomputingEnabled) {
|
|
|
|
// Set current maxTileSize for cache manager precomputation
|
|
|
|
const TRenderSettings &rs = renderDatas[0].m_info;
|
|
|
|
TPredictiveCacheManager::instance()->setMaxTileSize(rs.m_maxTileSize);
|
|
|
|
TPredictiveCacheManager::instance()->setBPP(rs.m_bpp);
|
|
|
|
|
|
|
|
// Perform the first precomputing run - fx usages declaration
|
|
|
|
{
|
|
|
|
locals::StatusDeclaration firstrunDecl(this, TRenderer::FIRSTRUN,
|
|
|
|
renderInfos);
|
|
|
|
|
|
|
|
for (kt = tasksVector.begin(); kt != kEnd; ++kt) {
|
|
|
|
if (hasToDie(renderId)) return;
|
2016-03-19 06:57:51 +13:00
|
|
|
|
2016-06-15 18:43:10 +12:00
|
|
|
(*kt)->preRun();
|
|
|
|
|
|
|
|
// NOTE: Thread-specific data must be temporarily uninstalled before
|
|
|
|
// processing events (which may redefine the thread data).
|
|
|
|
locals::clearStorage();
|
|
|
|
QCoreApplication::instance()->processEvents();
|
|
|
|
locals::setStorage(this, renderId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pass to the TESTRUN status - this one should faithfully reproduce
|
|
|
|
// the actual COMPUTING status
|
|
|
|
{
|
|
|
|
locals::StatusDeclaration testrunDecl(this, TRenderer::TESTRUN,
|
|
|
|
renderInfos);
|
|
|
|
|
|
|
|
for (kt = tasksVector.begin(); kt != kEnd; ++kt) {
|
|
|
|
if (hasToDie(renderId)) return;
|
|
|
|
|
|
|
|
(*kt)->preRun();
|
|
|
|
|
|
|
|
// NOTE: Thread-specific data must be temporarily uninstalled before
|
|
|
|
// processing events (which may redefine the thread data).
|
|
|
|
locals::clearStorage();
|
|
|
|
QCoreApplication::instance()->processEvents();
|
|
|
|
locals::setStorage(this, renderId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// Render
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
locals::declareStatusStart(this, TRenderer::COMPUTING, renderInfos);
|
|
|
|
|
|
|
|
// Update the tasks counts
|
|
|
|
m_undoneTasks += tasksVector.size();
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_renderInstancesMutex);
|
|
|
|
renderInfos->m_activeTasks = tasksVector.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
renderDecl.commit(); // Declarations are taken over by render tasks
|
|
|
|
}
|
|
|
|
|
|
|
|
instanceDecl.commit(); // Same here
|
|
|
|
|
|
|
|
// Launch the render
|
|
|
|
for (kt = tasksVector.begin(); kt != tasksVector.end(); ++kt)
|
|
|
|
m_executor.addTask(*kt);
|
|
|
|
|
|
|
|
tasksVector.clear(); // Prevent tasks destruction by TasksCleaner
|
2016-03-19 06:57:51 +13:00
|
|
|
}
|
2016-06-15 18:43:10 +12:00
|
|
|
|
|
|
|
void TRenderer::initialize() { TRendererStartInvoker::instance(); }
|