9#include <QtCore/qthreadpool.h>
12#include <QtQuick/private/qsgcontext_p.h>
13#include <QtQuick/private/qsgrenderer_p.h>
15#include "graphobjects/qssgrenderroot_p.h"
16#include "graphobjects/qssgrenderlayer_p.h"
17#include "graphobjects/qssgrenderitem2d_p.h"
20#include "resourcemanager/qssgrenderbuffermanager_p.h"
23#include <condition_variable>
29static void reindexChildNodes(QSSGRenderNode &node,
const QSSGRenderNodeVersionType version, quint32 &dfsIdx, size_t &count)
31 Q_ASSERT(node.type != QSSGRenderNode::Type::Layer);
32 if (node.type != QSSGRenderNode::Type::Layer) {
36 if (node.h.version() != version)
37 node.h = QSSGRenderNodeHandle(0, version, dfsIdx);
38 for (QSSGRenderNode &chld : node.children)
39 reindexChildNodes(chld, version, ++dfsIdx, ++count);
43static void reindexLayerChildNodes(QSSGRenderLayer &layer,
const QSSGRenderNodeVersionType version, quint32 &dfsIdx, size_t &count)
45 Q_ASSERT(layer.type == QSSGRenderNode::Type::Layer);
46 if (layer.type == QSSGRenderNode::Type::Layer) {
47 layer.h = QSSGRenderNodeHandle(0, version, 0);
48 for (QSSGRenderNode &chld : layer.children)
49 reindexChildNodes(chld, version, ++dfsIdx, ++count);
53static void reindex(QSSGRenderRoot *rootNode,
const QSSGRenderNodeVersionType version, quint32 &dfsIdx, size_t &count)
56 Q_ASSERT(rootNode->type == QSSGRenderNode::Type::Root);
57 Q_ASSERT(dfsIdx == 0);
58 rootNode->h = QSSGRenderNodeHandle(0, version, 0);
59 for (QSSGRenderNode &chld : rootNode->children)
60 reindexLayerChildNodes(
static_cast<QSSGRenderLayer &>(chld), version, dfsIdx, count);
68 Q_ASSERT_X(node.type != QSSGRenderNode::Type::Layer, Q_FUNC_INFO,
"Unexpected Layer node in child list!");
70 outList[idx++] = &node;
72 outList.push_back(&node);
73 for (QSSGRenderNode &chld : node.children)
74 collectChildNodesFirst<insert>(chld, outList, idx);
80 if (node.getGlobalState(QSSGRenderNode::GlobalState::Active)) {
82 outList[idx++] = &node;
84 outList.push_back(&node);
85 for (QSSGRenderNode &chld : node.children)
86 collectChildNodesSecond<insert>(chld, outList, idx);
94 Q_ASSERT(layer->type == QSSGRenderNode::Type::Layer);
96 for (QSSGRenderNode &chld : layer->children) {
97 if constexpr (discard == Discard::Inactive)
98 collectChildNodesSecond<insert>(chld, outList, idx);
100 collectChildNodesFirst<insert>(chld, outList, idx);
105 idx = outList.size();
110 const QSSGRenderNodeVersionType version,
111 QSSGGlobalRenderNodeData::GlobalTransformStore &globalTransforms,
112 QSSGGlobalRenderNodeData::GlobalOpacityStore &globalOpacities)
114 using DirtyFlag = QSSGRenderNode::DirtyFlag;
115 using FlagT = QSSGRenderNode::FlagT;
116 constexpr DirtyFlag TransformAndOpacityDirty = DirtyFlag(FlagT(DirtyFlag::TransformDirty) | FlagT(DirtyFlag::OpacityDirty));
118 if (Q_UNLIKELY(!node || (node->h.version() != version)))
122 bool retval = forcedRebuild || node->isDirty(TransformAndOpacityDirty);
128 const bool alwaysDirty = node->isDirty(DirtyFlag::StickyDirty);
130 if (retval || alwaysDirty) {
131 const auto idx = node->h.index();
133 auto &globalTransform = globalTransforms[idx];
134 auto &globalOpacity = globalOpacities[idx];
135 globalOpacity = node->localOpacity;
136 globalTransform = node->localTransform;
138 if (QSSGRenderNode *parent = node->parent) {
139 const auto pidx = parent->h.index();
140 const auto pOpacity = globalOpacities[pidx];
141 globalOpacity *= pOpacity;
143 if (parent->type != QSSGRenderGraphObject::Type::Layer) {
144 const auto pTransform = globalTransforms[pidx];
145 globalTransform = pTransform * node->localTransform;
149 node->clearDirty(TransformAndOpacityDirty);
159 using LocalState = QSSGRenderNode::LocalState;
160 using GlobalState = QSSGRenderNode::GlobalState;
161 using DirtyFlag = QSSGRenderNode::DirtyFlag;
162 using FlagT = QSSGRenderNode::FlagT;
164 constexpr DirtyFlag ClearDirtyMask = DirtyFlag(FlagT(DirtyFlag::ActiveDirty) | FlagT(DirtyFlag::PickableDirty) | FlagT(DirtyFlag::ImportDirty));
166 if (Q_UNLIKELY(!node || (node->h.version() != version)))
169 const bool activeDirty = node->isDirty(DirtyFlag::ActiveDirty);
170 const bool pickableDirty = node->isDirty(DirtyFlag::PickableDirty);
171 const bool importedDirty = node->isDirty(DirtyFlag::ImportDirty);
173 const bool updateState = activeDirty || pickableDirty || importedDirty;
176 const QSSGRenderNode *parent = node->parent;
177 const bool hasParent = (parent !=
nullptr);
178 const bool globallyActive = node->getLocalState(LocalState::Active) && (!hasParent || parent->getGlobalState(GlobalState::Active));
179 node->flags = globallyActive ? (node->flags | FlagT(GlobalState::Active)) : (node->flags & ~FlagT(GlobalState::Active));
180 const bool globallyPickable = node->getLocalState(LocalState::Pickable) || (hasParent && parent->getGlobalState(GlobalState::Pickable));
181 node->flags = globallyPickable ? (node->flags | FlagT(GlobalState::Pickable)) : (node->flags & ~FlagT(GlobalState::Pickable));
182 const bool globallyImported = node->getLocalState(LocalState::Imported) || (hasParent && parent->getGlobalState(GlobalState::Imported));
183 node->flags = globallyImported ? (node->flags | FlagT(GlobalState::Imported)) : (node->flags & ~FlagT(GlobalState::Imported));
186 node->clearDirty(ClearDirtyMask);
193 const VersionType version,
194 QSSGGlobalRenderNodeData::GlobalTransformStore &globalTransforms,
195 QSSGGlobalRenderNodeData::InstanceTransformStore &instanceTransforms)
197 if (Q_UNLIKELY(!node || (node->h.version() != version)))
200 constexpr bool retval =
true;
205 const auto idx = node->h.index();
206 QSSGRenderNode *parent = node->parent;
207 if (parent && parent->type != QSSGRenderGraphObject::Type::Layer && node->getLocalState(QSSGRenderNode::LocalState::Active)) {
208 const auto pidx = parent->h.index();
209 const auto &pGlobalTransform = globalTransforms[pidx];
210 QSSGRenderNode *instanceRoot = node->instanceRoot;
211 if (instanceRoot == node) {
212 instanceTransforms[idx] = { node->localTransform, pGlobalTransform };
213 }
else if (instanceRoot) {
214 auto &[nodeInstanceLocalTransform, nodeInstanceGlobalTransform] = instanceTransforms[idx];
215 auto &[instanceRootLocalTransform, instanceRootGlobalTransform] = instanceTransforms[instanceRoot->h.index()];
216 nodeInstanceGlobalTransform = instanceRootGlobalTransform;
219 nodeInstanceLocalTransform = node->localTransform;
222 if (p == instanceRoot) {
223 nodeInstanceLocalTransform = instanceRootLocalTransform * nodeInstanceLocalTransform;
226 nodeInstanceLocalTransform = p->localTransform * nodeInstanceLocalTransform;
232 QMatrix4x4 globalInstanceTransform = globalTransforms[pidx];
233 QMatrix4x4 localInstanceTransform = node->localTransform;
234 auto &localInstanceMatrix = *
reinterpret_cast<
float (*)[4][4]>(localInstanceTransform.data());
235 QVector3D localPos{localInstanceMatrix[3][0], localInstanceMatrix[3][1], localInstanceMatrix[3][2]};
236 localInstanceMatrix[3][0] = 0;
237 localInstanceMatrix[3][1] = 0;
238 localInstanceMatrix[3][2] = 0;
239 globalInstanceTransform = pGlobalTransform;
240 globalInstanceTransform.translate(localPos);
241 instanceTransforms[idx] = { localInstanceTransform, globalInstanceTransform };
244 instanceTransforms[idx] = { node->localTransform, {} };
250QSSGGlobalRenderNodeData::QSSGGlobalRenderNodeData(QSSGRenderRoot *root)
251#if QT_CONFIG(thread) && !defined(Q_OS_WASM)
252 : m_threadPool(
new QThreadPool)
261QSSGGlobalRenderNodeData::~QSSGGlobalRenderNodeData()
266void QSSGGlobalRenderNodeData::reindex()
278 if (Q_UNLIKELY(m_version == 0)) {
279 qCDebug(lcLayerData) <<
"Version wrap around detected, resetting to 1.";
283 ::reindex(m_rootNode, m_version, dfsIdx, m_nodeCount);
291 globalTransforms.resize(m_size, QMatrix4x4{ Qt::Uninitialized });
292 globalOpacities.resize(m_size, 1.0f);
293 instanceTransforms.resize(m_size, { QMatrix4x4{ Qt::Uninitialized }, QMatrix4x4{ Qt::Uninitialized } });
295 collectNodes(m_rootNode);
302void QSSGGlobalRenderNodeData::invalidate()
304 m_rootNode =
nullptr;
307QMatrix4x4 QSSGGlobalRenderNodeData::getGlobalTransform(QSSGRenderNodeHandle h, QMatrix4x4 defaultValue)
const
310 const bool hasId = h.hasId();
311 const bool validVersion = hasId && (h.version() == m_version);
312 const auto index = h.index();
316 if (!validVersion || !(globalTransforms.size() > index))
319 return globalTransforms[index];
322QMatrix4x4 QSSGGlobalRenderNodeData::getGlobalTransform(QSSGRenderNodeHandle h)
const
324 return getGlobalTransform(h, QMatrix4x4{ Qt::Uninitialized });
327QMatrix4x4 QSSGGlobalRenderNodeData::getGlobalTransform(
const QSSGRenderNode &node)
const
329 return getGlobalTransform(node.h, node.localTransform);
332float QSSGGlobalRenderNodeData::getGlobalOpacity(QSSGRenderNodeHandle h,
float defaultValue)
const
334 const bool hasId = h.hasId();
335 const bool validVersion = hasId && (h.version() == m_version);
336 const auto index = h.index();
338 if (!validVersion || !(globalOpacities.size() > index))
341 return globalOpacities[index];
344float QSSGGlobalRenderNodeData::getGlobalOpacity(
const QSSGRenderNode &node)
const
346 return getGlobalOpacity(node.h, node.localOpacity);
351QThreadPool *QSSGGlobalRenderNodeData::threadPool()
const {
return nullptr; }
353QThreadPool *QSSGGlobalRenderNodeData::threadPool()
const {
return m_threadPool.get(); }
357QSSGGlobalRenderNodeData::LayerNodeView QSSGGlobalRenderNodeData::getLayerNodeView(QSSGRenderLayerHandle h)
const
359 const bool hasId = h.hasId();
360 const bool validVersion = hasId && (h.version() == m_version);
361 const auto index = h.index();
363 if (!validVersion || !(layerNodes.size() > index))
366 auto §ion = layerNodes[index];
368 return { nodes.data() + section.offset, qsizetype(section.size) };
371QSSGGlobalRenderNodeData::LayerNodeView QSSGGlobalRenderNodeData::getLayerNodeView(
const QSSGRenderLayer &layer)
const
373 return getLayerNodeView(layer.lh);
376void QSSGGlobalRenderNodeData::collectNodes(QSSGRenderRoot *rootNode)
381 Q_ASSERT(rootNode !=
nullptr);
384 nodes.resize(m_nodeCount,
nullptr);
388 quint32 layerIdx = 0;
389 for (QSSGRenderNode &chld : rootNode->children) {
390 Q_ASSERT(chld.type == QSSGRenderNode::Type::Layer);
391 QSSGRenderLayer *layer =
static_cast<QSSGRenderLayer *>(&chld);
392 const size_t offset = idx;
393 collectLayerChildNodes<Discard::None, Insert::Indexed>(layer, nodes, idx);
394 layer->lh = QSSGRenderLayerHandle(layer->h.context(), m_version, layerIdx++);
395 layerNodes.emplace_back(LayerNodeSection{offset , idx - offset});
401void QSSGGlobalRenderNodeData::updateGlobalState()
404 for (QSSGRenderNode *node : nodes)
405 QSSGRenderDataHelpers::updateGlobalNodeState(node, m_version);
408 for (QSSGRenderNode *node : nodes)
409 calcGlobalNodeDataIndexedImpl<QSSGRenderDataHelpers::Strategy::Initial>(node, m_version, globalTransforms, globalOpacities);
412 for (QSSGRenderNode *node : nodes)
413 QSSGRenderDataHelpers::calcInstanceTransforms(node, m_version, globalTransforms, instanceTransforms);
416QSSGGlobalRenderNodeData::InstanceTransforms QSSGGlobalRenderNodeData::getInstanceTransforms(
const QSSGRenderNode &node)
const
418 return getInstanceTransforms(node.h);
421QSSGGlobalRenderNodeData::InstanceTransforms QSSGGlobalRenderNodeData::getInstanceTransforms(QSSGRenderNodeHandle h)
const
423 const bool hasId = h.hasId();
424 const bool validVersion = hasId && (h.version() == m_version);
425 const auto index = h.index();
427 if (!validVersion || !(instanceTransforms.size() > index))
430 return instanceTransforms[index];
434 : m_gnd(globalNodeData)
435 , m_version(globalNodeData->version())
442 const bool hasId = h.hasId();
443 const bool validVersion = hasId && (h.version() == m_version);
444 const auto index = h.index();
445 if (!validVersion || !(normalMatrices.size() > index))
448 return normalMatrices[index];
453 return getNormalMatrix(model.mh, QMatrix3x3{ Qt::Uninitialized });
458 const bool hasId = h.hasId();
459 const bool validVersion = hasId && (h.version() == m_version);
460 const auto index = h.index();
462 if (!validVersion || !(meshes.size() > index))
465 return meshes[index];
470 return getMesh(model.mh);
475 const bool hasId = h.hasId();
476 const bool validVersion = hasId && (h.version() == m_version);
477 const auto index = h.index();
479 if (!validVersion || !(materials.size() > index))
482 return materials[index];
487 return getMaterials(model.mh);
490QSSGRenderModelData::ModelViewProjections
QSSGRenderModelData::getModelViewProjection(
const QSSGRenderModel &model)
const
492 return getModelViewProjection(model.mh);
497 const bool hasId = h.hasId();
498 const bool validVersion = hasId && (h.version() == m_version);
499 const auto index = h.index();
501 if (!validVersion || !(modelViewProjections.size() > index))
504 return modelViewProjections[index];
509 const auto &bufferManager = renderer->contextInterface()->bufferManager();
511 const bool globalPickingEnabled = QSSGRendererPrivate::isGlobalPickingEnabled(*renderer);
513 for (
auto *model : models) {
522 if (
auto *theMesh = bufferManager->loadMesh(*model)) {
523 meshes[model->mh.index()] = theMesh;
527 const float modelGlobalOpacity = m_gnd->getGlobalOpacity(*model);
528 const bool canModelBePickable = (modelGlobalOpacity > QSSGRendererPrivate::minimumRenderOpacity)
529 && (globalPickingEnabled
530 || model->getGlobalState(QSSGRenderModel::GlobalState::Pickable));
531 if (canModelBePickable) {
534 const QSSGMesh::Mesh mesh = bufferManager->loadLightmapMesh(*model);
537 theMesh->bvh = bufferManager->loadMeshBVH(mesh);
538 else if (model->geometry)
539 theMesh->bvh = bufferManager->loadMeshBVH(model->geometry);
540 else if (!model->meshPath.isNull())
541 theMesh->bvh = bufferManager->loadMeshBVH(model->meshPath);
544 const auto &roots = theMesh->bvh->roots();
545 for (qsizetype i = 0, end = qsizetype(roots.size()); i < end; ++i)
546 theMesh->subsets[i].bvhRoot = roots[i];
551 const size_t index = model->mh.index();
552 if (QSSG_GUARD(meshes.size() > index))
553 meshes[model->mh.index()] =
nullptr;
561 bufferManager->commitBufferResourceUpdates();
566 for (
auto *model : models) {
567 const size_t index = model->mh.index();
568 if (QSSG_GUARD(materials.size() > index))
569 materials[index] = model->materials;
575 const auto modelCount = size_t(models.size());
576 const bool versionChanged = m_version != m_gnd->version();
577 const bool storageSizeChanged = (normalMatrices.size() < modelCount);
582 const bool reIndexNeeded = versionChanged || storageSizeChanged ||
true;
584 const QMatrix3x3 defaultNormalMatrix;
585 const QMatrix4x4 defaultModelViewProjection;
588 modelViewProjections.resize(modelCount, { defaultModelViewProjection, defaultModelViewProjection });
589 normalMatrices.resize(modelCount, defaultNormalMatrix);
590 meshes.resize(modelCount,
nullptr);
591 materials.resize(modelCount, {});
595 m_version = m_gnd->version();
597 for (quint32 i = 0; i < modelCount; ++i) {
598 QSSGRenderModel *model = models[i];
599 model->mh = QSSGRenderModelHandle(model->h.context(), model->h.version(), i);
603#if QT_CONFIG(thread) && !defined(Q_OS_WASM)
604 bool matrixesDone =
false;
605 bool mvpsDone =
false;
607 std::condition_variable cv;
611 const auto doNormalMatrices = [&]() {
612 for (
const QSSGRenderModel *model : std::as_const(models)) {
613 auto &normalMatrix = normalMatrices[model->mh.index()];
614 const QMatrix4x4 globalTransform = m_gnd->getGlobalTransform(*model);
615 QSSGRenderNode::calculateNormalMatrix(globalTransform, normalMatrix);
617#if QT_CONFIG(thread) && !defined(Q_OS_WASM)
618 std::lock_guard<std::mutex> guard(mutex);
625 const auto doMVPs = [&]() {
626 for (
const QSSGRenderModel *model : std::as_const(models)) {
628 const QMatrix4x4 globalTransform = m_gnd->getGlobalTransform(*model);
629 auto &mvp = modelViewProjections[model->mh.index()];
630 for (
const QSSGRenderCameraData &cameraData : renderCameraData)
631 QSSGRenderNode::calculateMVP(globalTransform, cameraData.viewProjection, mvp[mvpCount++]);
633#if QT_CONFIG(thread) && !defined(Q_OS_WASM)
634 std::lock_guard<std::mutex> guard(mutex);
641 auto *threadPool = m_gnd->threadPool();
642#define qssgTryThreadedStart(func)
645 } else if (!threadPool->tryStart(func)) {
646 qWarning("Unable to start thread for %s!", #func);
650#define qssgTryThreadedStart(func)
659 prepareMaterials(models);
660 prepareMeshData(models, renderer);
663#if QT_CONFIG(thread) && !defined(Q_OS_WASM)
664 std::unique_lock<std::mutex> guard(mutex);
665 cv.wait(guard, [&]() {
return matrixesDone && mvpsDone; });
669bool QSSGRenderDataHelpers::updateGlobalNodeDataIndexed(QSSGRenderNode *node,
const VersionType version, QSSGGlobalRenderNodeData::GlobalTransformStore &globalTransforms, QSSGGlobalRenderNodeData::GlobalOpacityStore &globalOpacities)
671 return calcGlobalNodeDataIndexedImpl<
Strategy::Update>(node, version, globalTransforms, globalOpacities);
674bool QSSGRenderDataHelpers::calcGlobalVariablesIndexed(QSSGRenderNode *node,
const VersionType version, QSSGGlobalRenderNodeData::GlobalTransformStore &globalTransforms, QSSGGlobalRenderNodeData::GlobalOpacityStore &globalOpacities)
676 return calcGlobalNodeDataIndexedImpl<
Strategy::Initial>(node, version, globalTransforms, globalOpacities);
681 const auto foundIt = item2DRenderers.find(&item);
682 return (foundIt != item2DRenderers.cend()) ? foundIt->second : Item2DRenderer{};
687 const bool hasId = h.hasId();
688 const bool validVersion = hasId && (h.version() == m_version);
689 const auto index = h.index();
691 if (!validVersion || !(modelViewProjections.size() > index))
694 return modelViewProjections[index];
699 return getModelViewProjection(item.ih);
704 m_gnd = globalNodeData;
705 m_version = globalNodeData->version();
715 const auto itemCount = size_t(items.size());
720 const bool versionChanged = m_version != m_gnd->version();
721 const bool storageSizeChanged = (modelViewProjections.size() < itemCount);
725 const bool reIndexNeeded = versionChanged || storageSizeChanged ||
true;
727 const QMatrix4x4 defaultModelViewProjection;
729 const auto &rhiCtx = renderer->contextInterface()->rhiContext();
732 modelViewProjections.resize(itemCount, { defaultModelViewProjection, defaultModelViewProjection });
736 m_version = m_gnd->version();
738 for (quint32 i = 0; i < itemCount; ++i) {
739 QSSGRenderItem2D *item = items[i];
740 item->ih = QSSGRenderItem2DHandle(item->h.context(), item->h.version(), i);
745 const auto &clipSpaceCorrMatrix = rhiCtx->rhi()->clipSpaceCorrMatrix();
748 const auto doMVPs = [&]() {
749 for (
const QSSGRenderItem2D *item : std::as_const(items)) {
751 const QMatrix4x4 globalTransform = m_gnd->getGlobalTransform(*item);
752 auto &mvps = modelViewProjections[item->ih.index()];
753 for (
const QSSGRenderCameraData &cameraData : renderCameraData) {
754 const QMatrix4x4 &mvp = cameraData.viewProjection * globalTransform;
755 mvps[mvpCount++] = clipSpaceCorrMatrix * mvp * flipMatrix;
764 QSGRenderContext *sgRc = QSSGRendererPrivate::getSgRenderContext(*renderer);
765 const bool contextChanged = (item2DRenderContext && item2DRenderContext != sgRc);
766 item2DRenderContext = sgRc;
768 for (
const QSSGRenderItem2D *theItem2D : std::as_const(items)) {
769 auto item2DRenderer = getItem2DRenderer(*theItem2D);
770 if (contextChanged) {
771 delete item2DRenderer;
774 item2DRenderer.clear();
777 if (!item2DRenderer) {
778 item2DRenderer = sgRc->createRenderer(QSGRendererInterface::RenderMode3D);
779 QObject::connect(item2DRenderer, SIGNAL(sceneGraphChanged()), theItem2D->m_frontEndObject, SLOT(update()));
782 if (item2DRenderer->rootNode() != theItem2D->m_rootNode) {
783 item2DRenderer->setRootNode(theItem2D->m_rootNode);
784 theItem2D->m_rootNode->markDirty(QSGNode::DirtyForceUpdate);
785 item2DRenderer->nodeChanged(theItem2D->m_rootNode, QSGNode::DirtyForceUpdate);
788 item2DRenderers[theItem2D] = item2DRenderer;
794 const auto foundIt = item2DRenderers.find(&item);
795 if (foundIt != item2DRenderers.cend()) {
796 delete foundIt->second;
797 item2DRenderers.erase(foundIt);
806 for (
auto &it : item2DRenderers)
808 item2DRenderers.clear();
809 item2DRenderContext =
nullptr;
810 modelViewProjections.clear();
friend class QSSGRenderer
void releaseRenderData(const QSSGRenderItem2D &item)
void updateItem2DData(QSSGItem2DsView &items, QSSGRenderer *renderer, const QSSGRenderCameraDataList &renderCameraData)
Item2DRenderer getItem2DRenderer(const QSSGRenderItem2D &item) const
ModelViewProjections getModelViewProjection(QSSGRenderItem2DHandle h) const
ModelViewProjections getModelViewProjection(const QSSGRenderItem2D &item) const
void updateModelData(QSSGModelsView &models, QSSGRenderer *renderer, const QSSGRenderCameraDataList &renderCameraData)
QMatrix3x3 getNormalMatrix(QSSGRenderModelHandle h, QMatrix3x3 defaultValue) const
QSSGRenderMesh * getMesh(QSSGRenderModelHandle h) const
MaterialList getMaterials(QSSGRenderModelHandle h) const
ModelViewProjections getModelViewProjection(QSSGRenderModelHandle h) const
QT_BEGIN_NAMESPACE Q_STATIC_LOGGING_CATEGORY(lcSynthesizedIterableAccess, "qt.iterable.synthesized", QtWarningMsg)
static bool calcGlobalNodeDataIndexedImpl(QSSGRenderNode *node, const QSSGRenderNodeVersionType version, QSSGGlobalRenderNodeData::GlobalTransformStore &globalTransforms, QSSGGlobalRenderNodeData::GlobalOpacityStore &globalOpacities)
static void reindex(QSSGRenderRoot *rootNode, const QSSGRenderNodeVersionType version, quint32 &dfsIdx, size_t &count)
static void reindexLayerChildNodes(QSSGRenderLayer &layer, const QSSGRenderNodeVersionType version, quint32 &dfsIdx, size_t &count)
static QT_BEGIN_NAMESPACE void reindexChildNodes(QSSGRenderNode &node, const QSSGRenderNodeVersionType version, quint32 &dfsIdx, size_t &count)
static void collectLayerChildNodes(QSSGRenderLayer *layer, QSSGGlobalRenderNodeData::NodeStore &outList, size_t &idx)
static void collectChildNodesFirst(QSSGRenderNode &node, QSSGGlobalRenderNodeData::NodeStore &outList, size_t &idx)
#define qssgTryThreadedStart(func)
static void collectChildNodesSecond(QSSGRenderNode &node, QSSGGlobalRenderNodeData::NodeStore &outList, size_t &idx)