renderProbeMgr.cpp 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2012 GarageGames, LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. #include "renderProbeMgr.h"
  23. #include "console/consoleTypes.h"
  24. #include "scene/sceneObject.h"
  25. #include "materials/materialManager.h"
  26. #include "scene/sceneRenderState.h"
  27. #include "math/util/sphereMesh.h"
  28. #include "math/util/matrixSet.h"
  29. #include "materials/processedMaterial.h"
  30. #include "renderInstance/renderDeferredMgr.h"
  31. #include "math/mPolyhedron.impl.h"
  32. #include "gfx/gfxTransformSaver.h"
  33. #include "gfx/gfxDebugEvent.h"
  34. #include "shaderGen/shaderGenVars.h"
  35. #include "materials/shaderData.h"
  36. #include "gfx/gfxTextureManager.h"
  37. #include "postFx/postEffect.h"
  38. #include "T3D/lighting/reflectionProbe.h"
  39. #include "T3D/lighting/IBLUtilities.h"
  40. //For our cameraQuery setup
  41. #include "T3D/gameTSCtrl.h"
  42. #define TORQUE_GFX_VISUAL_DEBUG //renderdoc debugging
  43. IMPLEMENT_CONOBJECT(RenderProbeMgr);
  44. ConsoleDocClass( RenderProbeMgr,
  45. "@brief A render bin which uses object callbacks for rendering.\n\n"
  46. "This render bin gathers object render instances and calls its delegate "
  47. "method to perform rendering. It is used infrequently for specialized "
  48. "scene objects which perform custom rendering.\n\n"
  49. "@ingroup RenderBin\n" );
  50. RenderProbeMgr *RenderProbeMgr::smProbeManager = NULL;
  51. bool RenderProbeMgr::smRenderReflectionProbes = true;
  52. S32 QSORT_CALLBACK AscendingReflectProbeInfluence(const void* a, const void* b)
  53. {
  54. // Debug Profiling.
  55. PROFILE_SCOPE(AdvancedLightBinManager_AscendingReflectProbeInfluence);
  56. // Fetch asset definitions.
  57. const ProbeRenderInst* pReflectProbeA = (*(ProbeRenderInst**)a);
  58. const ProbeRenderInst* pReflectProbeB = (*(ProbeRenderInst**)b);
  59. //sort by score
  60. return pReflectProbeA->mScore - pReflectProbeB->mScore;
  61. }
  62. //
  63. //
  64. ProbeRenderInst::ProbeRenderInst() : SystemInterface(),
  65. mTransform(true),
  66. mDirty(false),
  67. mPriority(1.0f),
  68. mScore(0.0f),
  69. mPrefilterCubemap(NULL),
  70. mIrradianceCubemap(NULL),
  71. mRadius(1.0f),
  72. mProbeRefOffset(0, 0, 0),
  73. mProbeRefScale(1,1,1),
  74. mAtten(0.0),
  75. mCubemapIndex(0),
  76. mIsSkylight(false)
  77. {
  78. }
  79. ProbeRenderInst::~ProbeRenderInst()
  80. {
  81. if (mPrefilterCubemap && mPrefilterCubemap.isValid())
  82. {
  83. mPrefilterCubemap.free();
  84. }
  85. if (mIrradianceCubemap && mIrradianceCubemap.isValid())
  86. {
  87. mIrradianceCubemap.free();
  88. }
  89. }
  90. void ProbeRenderInst::set(const ProbeRenderInst *probeInfo)
  91. {
  92. mTransform = probeInfo->mTransform;
  93. mPrefilterCubemap = probeInfo->mPrefilterCubemap;
  94. mIrradianceCubemap = probeInfo->mIrradianceCubemap;
  95. mRadius = probeInfo->mRadius;
  96. mProbeShapeType = probeInfo->mProbeShapeType;
  97. mBounds = probeInfo->mBounds;
  98. mIsSkylight = probeInfo->mIsSkylight;
  99. mScore = probeInfo->mScore;
  100. mAtten = probeInfo->mAtten;
  101. }
  102. //
  103. //
  104. ProbeShaderConstants::ProbeShaderConstants()
  105. : mInit(false),
  106. mShader(NULL),
  107. mProbeParamsSC(NULL),
  108. mProbePositionSC(NULL),
  109. mProbeRefPosSC(NULL),
  110. mProbeBoxMinSC(NULL),
  111. mProbeBoxMaxSC(NULL),
  112. mProbeConfigDataSC(NULL),
  113. mProbeSpecularCubemapSC(NULL),
  114. mProbeIrradianceCubemapSC(NULL),
  115. mProbeCountSC(NULL)
  116. {
  117. }
  118. ProbeShaderConstants::~ProbeShaderConstants()
  119. {
  120. if (mShader.isValid())
  121. {
  122. mShader->getReloadSignal().remove(this, &ProbeShaderConstants::_onShaderReload);
  123. mShader = NULL;
  124. }
  125. }
  126. void ProbeShaderConstants::init(GFXShader* shader)
  127. {
  128. if (mShader.getPointer() != shader)
  129. {
  130. if (mShader.isValid())
  131. mShader->getReloadSignal().remove(this, &ProbeShaderConstants::_onShaderReload);
  132. mShader = shader;
  133. mShader->getReloadSignal().notify(this, &ProbeShaderConstants::_onShaderReload);
  134. }
  135. mProbeParamsSC = shader->getShaderConstHandle("$probeParams");
  136. //Reflection Probes
  137. mProbePositionSC = shader->getShaderConstHandle(ShaderGenVars::probePosition);
  138. mProbeRefPosSC = shader->getShaderConstHandle(ShaderGenVars::probeRefPos);
  139. mProbeBoxMinSC = shader->getShaderConstHandle(ShaderGenVars::probeBoxMin);
  140. mProbeBoxMaxSC = shader->getShaderConstHandle(ShaderGenVars::probeBoxMax);
  141. mWorldToObjArraySC = shader->getShaderConstHandle(ShaderGenVars::worldToObjArray);
  142. mProbeConfigDataSC = shader->getShaderConstHandle(ShaderGenVars::probeConfigData);
  143. mProbeSpecularCubemapSC = shader->getShaderConstHandle(ShaderGenVars::specularCubemapAR);
  144. mProbeIrradianceCubemapSC = shader->getShaderConstHandle(ShaderGenVars::irradianceCubemapAR);
  145. mProbeCountSC = shader->getShaderConstHandle(ShaderGenVars::probeCount);
  146. mSkylightPrefilterMap = shader->getShaderConstHandle(ShaderGenVars::skylightPrefilterMap);
  147. mSkylightIrradMap = shader->getShaderConstHandle(ShaderGenVars::skylightIrradMap);
  148. mHasSkylight = shader->getShaderConstHandle(ShaderGenVars::hasSkylight);
  149. mInit = true;
  150. }
  151. void ProbeShaderConstants::_onShaderReload()
  152. {
  153. if (mShader.isValid())
  154. init(mShader);
  155. }
  156. //
  157. //
  158. RenderProbeMgr::RenderProbeMgr()
  159. : RenderBinManager(RenderPassManager::RIT_Probes, 1.0f, 1.0f),
  160. mLastShader(nullptr),
  161. mLastConstants(nullptr)
  162. {
  163. mEffectiveProbeCount = 0;
  164. mMipCount = 0;
  165. mProbeArrayEffect = nullptr;
  166. smProbeManager = this;
  167. mCubeMapCount = 0;
  168. for (U32 i = 0; i < PROBE_MAX_COUNT; i++)
  169. {
  170. mCubeMapSlots[i] = false;
  171. }
  172. }
  173. RenderProbeMgr::RenderProbeMgr(RenderInstType riType, F32 renderOrder, F32 processAddOrder)
  174. : RenderBinManager(riType, renderOrder, processAddOrder)
  175. {
  176. }
  177. RenderProbeMgr::~RenderProbeMgr()
  178. {
  179. mLastShader = NULL;
  180. mLastConstants = NULL;
  181. for (ProbeConstantMap::Iterator i = mConstantLookup.begin(); i != mConstantLookup.end(); i++)
  182. {
  183. if (i->value)
  184. SAFE_DELETE(i->value);
  185. }
  186. mConstantLookup.clear();
  187. }
  188. bool RenderProbeMgr::onAdd()
  189. {
  190. if (!Parent::onAdd())
  191. return false;
  192. mIrradianceArray = GFXCubemapArrayHandle(GFX->createCubemapArray());
  193. mPrefilterArray = GFXCubemapArrayHandle(GFX->createCubemapArray());
  194. //pre-allocate a few slots
  195. mIrradianceArray->init(PROBE_ARRAY_SLOT_BUFFER_SIZE, PROBE_IRRAD_SIZE, PROBE_FORMAT);
  196. mPrefilterArray->init(PROBE_ARRAY_SLOT_BUFFER_SIZE, PROBE_PREFILTER_SIZE, PROBE_FORMAT);
  197. mCubeSlotCount = PROBE_ARRAY_SLOT_BUFFER_SIZE;
  198. //create our own default default skylight
  199. mDefaultSkyLight = new ProbeRenderInst;
  200. mDefaultSkyLight->mProbeShapeType = ProbeRenderInst::Skylight;
  201. if (!mDefaultSkyLight->mIrradianceCubemap.set("core/art/pbr/default_irradiance.dds"))
  202. {
  203. Con::errorf("RenderProbeMgr::onAdd: Failed to load default irradiance cubemap");
  204. return false;
  205. }
  206. if (!mDefaultSkyLight->mPrefilterCubemap.set("core/art/pbr/default_prefilter.dds"))
  207. {
  208. Con::errorf("RenderProbeMgr::onAdd: Failed to load default prefilter cubemap");
  209. return false;
  210. }
  211. return true;
  212. }
  213. void RenderProbeMgr::onRemove()
  214. {
  215. Parent::onRemove();
  216. }
  217. void RenderProbeMgr::initPersistFields()
  218. {
  219. Parent::initPersistFields();
  220. }
  221. void RenderProbeMgr::addElement(RenderInst *inst)
  222. {
  223. // If this instance is translucent handle it in RenderTranslucentMgr
  224. //if (inst->translucentSort)
  225. return;
  226. //AssertFatal(inst->defaultKey != 0, "RenderMeshMgr::addElement() - Got null sort key... did you forget to set it?");
  227. /*internalAddElement(inst);
  228. ProbeRenderInst* probeInst = static_cast<ProbeRenderInst*>(inst);
  229. if (probeInst->mIsSkylight)
  230. {
  231. addSkylightProbe(probeInst);
  232. }
  233. else
  234. {
  235. if (probeInst->mProbeShapeType == ProbeInfo::Sphere)
  236. addSphereReflectionProbe(probeInst);
  237. else
  238. addConvexReflectionProbe(probeInst);
  239. }*/
  240. }
  241. void RenderProbeMgr::registerProbe(U32 probeIdx)
  242. {
  243. //Mostly for consolidation, but also lets us sanity check or prep any other data we need for rendering this in one place at time of flagging for render
  244. if (probeIdx >= ProbeRenderInst::all.size())
  245. return;
  246. mRegisteredProbes.push_back_unique(probeIdx);
  247. if (!ProbeRenderInst::all[probeIdx]->mIsSkylight)
  248. {
  249. const U32 cubeIndex = _findNextEmptyCubeSlot();
  250. if (cubeIndex == INVALID_CUBE_SLOT)
  251. {
  252. Con::warnf("RenderProbeMgr::addProbe: Invalid cubemap slot.");
  253. return;
  254. }
  255. //check if we need to resize the cubemap array
  256. if (cubeIndex >= mCubeSlotCount)
  257. {
  258. //alloc temp array handles
  259. GFXCubemapArrayHandle irr = GFXCubemapArrayHandle(GFX->createCubemapArray());
  260. GFXCubemapArrayHandle prefilter = GFXCubemapArrayHandle(GFX->createCubemapArray());
  261. irr->init(mCubeSlotCount + PROBE_ARRAY_SLOT_BUFFER_SIZE, PROBE_IRRAD_SIZE, PROBE_FORMAT);
  262. prefilter->init(mCubeSlotCount + PROBE_ARRAY_SLOT_BUFFER_SIZE, PROBE_PREFILTER_SIZE, PROBE_FORMAT);
  263. mIrradianceArray->copyTo(irr);
  264. mPrefilterArray->copyTo(prefilter);
  265. //assign the temp handles to the new ones, this will destroy the old ones as well
  266. mIrradianceArray = irr;
  267. mPrefilterArray = prefilter;
  268. mCubeSlotCount += PROBE_ARRAY_SLOT_BUFFER_SIZE;
  269. }
  270. ProbeRenderInst::all[probeIdx]->mCubemapIndex = cubeIndex;
  271. //mark cubemap slot as taken
  272. mCubeMapSlots[cubeIndex] = true;
  273. mCubeMapCount++;
  274. Con::warnf("RenderProbeMgr::registerProbe: Registered probe %u to cubeIndex %u", probeIdx, cubeIndex);
  275. }
  276. //rebuild our probe data
  277. _setupStaticParameters();
  278. }
  279. void RenderProbeMgr::unregisterProbe(U32 probeIdx)
  280. {
  281. //Mostly for consolidation, but also lets us sanity check or prep any other data we need for rendering this in one place at time of flagging for render
  282. if (probeIdx >= ProbeRenderInst::all.size())
  283. return;
  284. mRegisteredProbes.remove(probeIdx);
  285. if (ProbeRenderInst::all[probeIdx]->mCubemapIndex == INVALID_CUBE_SLOT)
  286. return;
  287. //mark cubemap slot as available now
  288. mCubeMapSlots[ProbeRenderInst::all[probeIdx]->mCubemapIndex] = false;
  289. mCubeMapCount--;
  290. //rebuild our probe data
  291. _setupStaticParameters();
  292. }
  293. //
  294. //
  295. PostEffect* RenderProbeMgr::getProbeArrayEffect()
  296. {
  297. if (!mProbeArrayEffect)
  298. {
  299. mProbeArrayEffect = dynamic_cast<PostEffect*>(Sim::findObject("reflectionProbeArrayPostFX"));
  300. if (!mProbeArrayEffect)
  301. return nullptr;
  302. }
  303. return mProbeArrayEffect;
  304. }
  305. //remove
  306. //Con::setIntVariable("lightMetrics::activeReflectionProbes", mReflectProbeBin.size());
  307. //Con::setIntVariable("lightMetrics::culledReflectProbes", 0/*mNumLightsCulled*/);
  308. //
  309. void RenderProbeMgr::updateProbes()
  310. {
  311. _setupStaticParameters();
  312. }
  313. void RenderProbeMgr::_setupStaticParameters()
  314. {
  315. //Array rendering
  316. U32 probeCount = ProbeRenderInst::all.size();
  317. mEffectiveProbeCount = 0;
  318. mMipCount = 0;
  319. if (probePositionsData.size() != MAXPROBECOUNT)
  320. {
  321. probePositionsData.setSize(MAXPROBECOUNT);
  322. probeRefPositionsData.setSize(MAXPROBECOUNT);
  323. probeWorldToObjData.setSize(MAXPROBECOUNT);
  324. probeBBMinData.setSize(MAXPROBECOUNT);
  325. probeBBMaxData.setSize(MAXPROBECOUNT);
  326. probeConfigData.setSize(MAXPROBECOUNT);
  327. }
  328. probePositionsData.fill(Point4F::Zero);
  329. probeRefPositionsData.fill(Point4F::Zero);
  330. probeWorldToObjData.fill(MatrixF::Identity);
  331. probeBBMinData.fill(Point4F::Zero);
  332. probeBBMaxData.fill(Point4F::Zero);
  333. probeConfigData.fill(Point4F::Zero);
  334. cubeMaps.clear();
  335. irradMaps.clear();
  336. Vector<U32> cubemapIdxes;
  337. if (probeCount != 0 && ProbeRenderInst::all[0]->mPrefilterCubemap != nullptr)
  338. {
  339. //Get our mipCount
  340. mMipCount = ProbeRenderInst::all[0]->mPrefilterCubemap.getPointer()->getMipMapLevels();
  341. }
  342. else
  343. {
  344. mMipCount = 1;
  345. }
  346. for (U32 i = 0; i < probeCount; i++)
  347. {
  348. if (mEffectiveProbeCount >= MAXPROBECOUNT)
  349. break;
  350. const ProbeRenderInst& curEntry = *ProbeRenderInst::all[i];
  351. if (!curEntry.mIsEnabled)
  352. continue;
  353. if (curEntry.mIsSkylight)
  354. {
  355. skylightPos = curEntry.getPosition();
  356. skylightPrefilterMap = curEntry.mPrefilterCubemap;
  357. skylightIrradMap = curEntry.mIrradianceCubemap;
  358. hasSkylight = true;
  359. continue;
  360. }
  361. //Setup
  362. Point3F probePos = curEntry.getPosition();
  363. Point3F refPos = curEntry.getPosition() +curEntry.mProbeRefOffset;
  364. probePositionsData[mEffectiveProbeCount] = Point4F(probePos.x, probePos.y, probePos.z,0);
  365. probeRefPositionsData[mEffectiveProbeCount] = Point4F(refPos.x, refPos.y, refPos.z, 0);
  366. probeWorldToObjData[mEffectiveProbeCount] = curEntry.getTransform();
  367. Point3F bbMin = refPos - curEntry.mProbeRefScale/2 * curEntry.getTransform().getScale();
  368. Point3F bbMax = refPos + curEntry.mProbeRefScale/2 * curEntry.getTransform().getScale();
  369. probeBBMinData[mEffectiveProbeCount] = Point4F(bbMin.x, bbMin.y, bbMin.z, 0);
  370. probeBBMaxData[mEffectiveProbeCount] = Point4F(bbMax.x, bbMax.y, bbMax.z, 0);
  371. probeConfigData[mEffectiveProbeCount] = Point4F(curEntry.mProbeShapeType,
  372. curEntry.mRadius,
  373. curEntry.mAtten,
  374. curEntry.mCubemapIndex);
  375. cubeMaps.push_back(curEntry.mPrefilterCubemap);
  376. irradMaps.push_back(curEntry.mIrradianceCubemap);
  377. cubemapIdxes.push_back(i);
  378. mEffectiveProbeCount++;
  379. }
  380. /*if (mEffectiveProbeCount != 0)
  381. {
  382. bool useOldWay = false;
  383. if (useOldWay)
  384. {
  385. //old static way
  386. mPrefilterArray = GFXCubemapArrayHandle(GFX->createCubemapArray());
  387. mIrradianceArray = GFXCubemapArrayHandle(GFX->createCubemapArray());
  388. mPrefilterArray->init(cubeMaps.address(), cubeMaps.size());
  389. mIrradianceArray->init(irradMaps.address(), irradMaps.size());
  390. }
  391. else
  392. {
  393. //faked static way by doing it via update
  394. for (U32 i = 0; i < cubemapIdxes.size(); i++)
  395. {
  396. U32 probeIdx = cubemapIdxes[i];
  397. const U32 cubeIndex = ProbeRenderInst::all[probeIdx]->mCubemapIndex;
  398. mIrradianceArray->updateTexture(irradMaps[i], cubeIndex);
  399. mPrefilterArray->updateTexture(cubeMaps[i], cubeIndex);
  400. }
  401. }
  402. }*/
  403. }
  404. void RenderProbeMgr::updateProbeTexture(ProbeRenderInst* probe)
  405. {
  406. //We don't stuff skylights into the array, so we can just skip out on this if it's a skylight
  407. if (probe->mIsSkylight)
  408. return;
  409. S32 probeIdx = ProbeRenderInst::all.find_next(probe);
  410. if (probeIdx != -1) //i mean, the opposite shouldn't even be possible
  411. updateProbeTexture(probeIdx);
  412. }
  413. void RenderProbeMgr::updateProbeTexture(U32 probeIdx)
  414. {
  415. if (probeIdx >= ProbeRenderInst::all.size())
  416. return;
  417. const U32 cubeIndex = ProbeRenderInst::all[probeIdx]->mCubemapIndex;
  418. mIrradianceArray->updateTexture(ProbeRenderInst::all[probeIdx]->mIrradianceCubemap, cubeIndex);
  419. mPrefilterArray->updateTexture(ProbeRenderInst::all[probeIdx]->mPrefilterCubemap, cubeIndex);
  420. Con::warnf("UpdatedProbeTexture - probeIdx: %u on cubeIndex %u, Irrad validity: %d, Prefilter validity: %d", probeIdx, cubeIndex,
  421. ProbeRenderInst::all[probeIdx]->mIrradianceCubemap->isInitialized(), ProbeRenderInst::all[probeIdx]->mPrefilterCubemap->isInitialized());
  422. }
  423. void RenderProbeMgr::_setupPerFrameParameters(const SceneRenderState *state)
  424. {
  425. PROFILE_SCOPE(RenderProbeMgr_SetupPerFrameParameters);
  426. }
  427. ProbeShaderConstants* RenderProbeMgr::getProbeShaderConstants(GFXShaderConstBuffer* buffer)
  428. {
  429. if (!buffer)
  430. return NULL;
  431. PROFILE_SCOPE(ProbeManager_GetProbeShaderConstants);
  432. GFXShader* shader = buffer->getShader();
  433. // Check to see if this is the same shader, we'll get hit repeatedly by
  434. // the same one due to the render bin loops.
  435. if (mLastShader.getPointer() != shader)
  436. {
  437. ProbeConstantMap::Iterator iter = mConstantLookup.find(shader);
  438. if (iter != mConstantLookup.end())
  439. {
  440. mLastConstants = iter->value;
  441. }
  442. else
  443. {
  444. ProbeShaderConstants* psc = new ProbeShaderConstants();
  445. mConstantLookup[shader] = psc;
  446. mLastConstants = psc;
  447. }
  448. // Set our new shader
  449. mLastShader = shader;
  450. }
  451. // Make sure that our current lighting constants are initialized
  452. if (mLastConstants && !mLastConstants->mInit)
  453. mLastConstants->init(shader);
  454. return mLastConstants;
  455. }
  456. void RenderProbeMgr::_update4ProbeConsts(const SceneData &sgData,
  457. MatrixSet &matSet,
  458. ProbeShaderConstants *probeShaderConsts,
  459. GFXShaderConstBuffer *shaderConsts)
  460. {
  461. PROFILE_SCOPE(ProbeManager_Update4ProbeConsts);
  462. // Skip over gathering lights if we don't have to!
  463. if (probeShaderConsts->mProbePositionSC->isValid() ||
  464. probeShaderConsts->mProbeConfigDataSC->isValid() ||
  465. probeShaderConsts->mProbeBoxMinSC->isValid() ||
  466. probeShaderConsts->mProbeBoxMaxSC->isValid() ||
  467. probeShaderConsts->mProbeSpecularCubemapSC->isValid() ||
  468. probeShaderConsts->mProbeIrradianceCubemapSC->isValid()/* && (!ProbeRenderInst::all.empty())*/)
  469. {
  470. PROFILE_SCOPE(ProbeManager_Update4ProbeConsts_setProbes);
  471. static AlignedArray<Point3F> probePositions(4, sizeof(Point3F));
  472. static AlignedArray<F32> probeRadius(4, sizeof(F32));
  473. static AlignedArray<Point3F> probeBoxMins(4, sizeof(Point3F));
  474. static AlignedArray<Point3F> probeBoxMaxs(4, sizeof(Point3F));
  475. static AlignedArray<Point3F> probeLocalPositions(4, sizeof(Point3F));
  476. static AlignedArray<F32> probeIsSphere(4, sizeof(F32));
  477. //static AlignedArray<CubemapData> probeCubemap(4, sizeof(CubemapData));
  478. //F32 range;
  479. // Need to clear the buffers so that we don't leak
  480. // lights from previous passes or have NaNs.
  481. dMemset(probePositions.getBuffer(), 0, probePositions.getBufferSize());
  482. dMemset(probeRadius.getBuffer(), 0, probeRadius.getBufferSize());
  483. dMemset(probeBoxMins.getBuffer(), 0, probeBoxMins.getBufferSize());
  484. dMemset(probeBoxMaxs.getBuffer(), 0, probeBoxMaxs.getBufferSize());
  485. dMemset(probeLocalPositions.getBuffer(), 0, probeLocalPositions.getBufferSize());
  486. dMemset(probeIsSphere.getBuffer(), 0, probeRadius.getBufferSize());
  487. //dMemset(probeCubemap.getBuffer(), 0, probeCubemap.getBufferSize());
  488. matSet.restoreSceneViewProjection();
  489. //const MatrixF &worldToCameraXfm = matSet.getWorldToCamera();
  490. // Gather the data for the first 4 probes.
  491. /*const ProbeRenderInst *probe;
  492. for (U32 i = 0; i < 4; i++)
  493. {
  494. if (i >= ProbeRenderInst::all.size())
  495. break;
  496. probe = ProbeRenderInst::all[i];
  497. if (!probe)
  498. continue;
  499. if (!probe->mIsEnabled)
  500. continue;
  501. // The light positions and spot directions are
  502. // in SoA order to make optimal use of the GPU.
  503. const Point3F &probePos = probe->getPosition();
  504. probePositions[i].x = probePos.x;
  505. probePositions[i].y = probePos.y;
  506. probePositions[i].z = probePos.z;
  507. probeRadius[i] = probe->mRadius;
  508. const Point3F &minExt = probe->mBounds.minExtents;
  509. probeBoxMins[i].x = minExt.x;
  510. probeBoxMins[i].y = minExt.y;
  511. probeBoxMins[i].z = minExt.z;
  512. const Point3F &maxExt = probe->mBounds.maxExtents;
  513. probeBoxMaxs[i].x = maxExt.x;
  514. probeBoxMaxs[i].y = maxExt.y;
  515. probeBoxMaxs[i].z = maxExt.z;
  516. probeIsSphere[i] = probe->mProbeShapeType == ProbeRenderInst::Sphere ? 1.0 : 0.0;
  517. Point3F localProbePos;
  518. worldToCameraXfm.mulP(probe->getPosition(), &localProbePos);
  519. probeLocalPositions[i].x = localProbePos.x;
  520. probeLocalPositions[i].y = localProbePos.y;
  521. probeLocalPositions[i].z = localProbePos.z;
  522. if (probe->mCubemap && !probe->mCubemap.isNull())
  523. {
  524. S32 samplerReg = probeCubemapSC->getSamplerRegister();
  525. if (samplerReg != -1)
  526. GFX->setCubeTexture(samplerReg + i, probe->mCubemap.getPointer());
  527. }
  528. }*/
  529. //Array rendering
  530. U32 probeCount = ProbeRenderInst::all.size();
  531. mEffectiveProbeCount = 0;
  532. mMipCount = 0;
  533. if (probePositionsData.size() != MAXPROBECOUNT)
  534. {
  535. probePositionsData.setSize(MAXPROBECOUNT);
  536. probeRefPositionsData.setSize(MAXPROBECOUNT);
  537. probeWorldToObjData.setSize(MAXPROBECOUNT);
  538. probeBBMinData.setSize(MAXPROBECOUNT);
  539. probeBBMaxData.setSize(MAXPROBECOUNT);
  540. probeConfigData.setSize(MAXPROBECOUNT);
  541. }
  542. probePositionsData.fill(Point4F::Zero);
  543. probeRefPositionsData.fill(Point4F::Zero);
  544. probeWorldToObjData.fill(MatrixF::Identity);
  545. probeBBMinData.fill(Point4F::Zero);
  546. probeBBMaxData.fill(Point4F::Zero);
  547. probeConfigData.fill(Point4F::Zero);
  548. cubeMaps.clear();
  549. irradMaps.clear();
  550. Vector<U32> cubemapIdxes;
  551. U32 effectiveProbeCount = 0;
  552. bool hasSkylight = false;
  553. for (U32 i = 0; i < probeCount; i++)
  554. {
  555. if (effectiveProbeCount >= 4)
  556. break;
  557. const ProbeRenderInst& curEntry = *ProbeRenderInst::all[i];
  558. if (!curEntry.mIsEnabled)
  559. continue;
  560. if (curEntry.mIsSkylight)
  561. {
  562. GFX->setCubeTexture(probeShaderConsts->mSkylightPrefilterMap->getSamplerRegister(), curEntry.mPrefilterCubemap);
  563. GFX->setCubeTexture(probeShaderConsts->mSkylightIrradMap->getSamplerRegister(), curEntry.mIrradianceCubemap);
  564. shaderConsts->setSafe(probeShaderConsts->mHasSkylight, true);
  565. hasSkylight = true;
  566. continue;
  567. }
  568. //Setup
  569. /*Point3F probePos = curEntry.getPosition();
  570. Point3F refPos = curEntry.getPosition() + curEntry.mProbeRefOffset;
  571. probePositionsData[mEffectiveProbeCount] = Point4F(probePos.x, probePos.y, probePos.z, 0);
  572. probeRefPositionsData[mEffectiveProbeCount] = Point4F(refPos.x, refPos.y, refPos.z, 0);
  573. probeWorldToObjData[mEffectiveProbeCount] = curEntry.getTransform();
  574. Point3F bbMin = refPos - curEntry.mProbeRefScale / 2 * curEntry.getTransform().getScale();
  575. Point3F bbMax = refPos + curEntry.mProbeRefScale / 2 * curEntry.getTransform().getScale();
  576. probeBBMinData[mEffectiveProbeCount] = Point4F(bbMin.x, bbMin.y, bbMin.z, 0);
  577. probeBBMaxData[mEffectiveProbeCount] = Point4F(bbMax.x, bbMax.y, bbMax.z, 0);
  578. probeConfigData[mEffectiveProbeCount] = Point4F(curEntry.mProbeShapeType,
  579. curEntry.mRadius,
  580. curEntry.mAtten,
  581. curEntry.mCubemapIndex);
  582. cubeMaps.push_back(curEntry.mPrefilterCubemap);
  583. irradMaps.push_back(curEntry.mIrradianceCubemap);
  584. cubemapIdxes.push_back(i);*/
  585. effectiveProbeCount++;
  586. }
  587. if (!hasSkylight)
  588. shaderConsts->setSafe(probeShaderConsts->mHasSkylight, false);
  589. }
  590. /*else
  591. {
  592. if (probeCubemapSC->isValid())
  593. {
  594. for (U32 i = 0; i < 4; ++i)
  595. GFX->setCubeTexture(probeCubemapSC->getSamplerRegister() + i, NULL);
  596. }
  597. }*/
  598. }
  599. void RenderProbeMgr::setProbeInfo(ProcessedMaterial *pmat,
  600. const Material *mat,
  601. const SceneData &sgData,
  602. const SceneRenderState *state,
  603. U32 pass,
  604. GFXShaderConstBuffer *shaderConsts)
  605. {
  606. // Skip this if we're rendering from the deferred bin.
  607. if (sgData.binType == SceneData::DeferredBin)
  608. return;
  609. // if (mRegisteredProbes.empty())
  610. // return;
  611. PROFILE_SCOPE(ProbeManager_setProbeInfo);
  612. ProbeShaderConstants *psc = getProbeShaderConstants(shaderConsts);
  613. // NOTE: If you encounter a crash from this point forward
  614. // while setting a shader constant its probably because the
  615. // mConstantLookup has bad shaders/constants in it.
  616. //
  617. // This is a known crash bug that can occur if materials/shaders
  618. // are reloaded and the light manager is not reset.
  619. //
  620. // We should look to fix this by clearing the table.
  621. MatrixSet matSet = state->getRenderPass()->getMatrixSet();
  622. // Update the forward shading light constants.
  623. _update4ProbeConsts(sgData, matSet, psc, shaderConsts);
  624. }
  625. //-----------------------------------------------------------------------------
  626. // render objects
  627. //-----------------------------------------------------------------------------
  628. void RenderProbeMgr::render( SceneRenderState *state )
  629. {
  630. //PROFILE_SCOPE(RenderProbeMgr_render);
  631. if (getProbeArrayEffect() == nullptr)
  632. return;
  633. //updateProbes();
  634. // Early out if nothing to draw.
  635. if (!ProbeRenderInst::all.size() || !RenderProbeMgr::smRenderReflectionProbes || !state->isDiffusePass() || (mEffectiveProbeCount == 0
  636. || mCubeMapCount != 0 && !hasSkylight))
  637. {
  638. getProbeArrayEffect()->setSkip(true);
  639. return;
  640. }
  641. GFXTransformSaver saver;
  642. GFXDEBUGEVENT_SCOPE(RenderProbeMgr_render, ColorI::WHITE);
  643. // Initialize and set the per-frame parameters after getting
  644. // the vector light material as we use lazy creation.
  645. //_setupPerFrameParameters(state);
  646. //Array rendering
  647. //U32 probeCount = ProbeRenderInst::all.size();
  648. mProbeArrayEffect->setShaderConst("$hasSkylight", (float)hasSkylight);
  649. if (hasSkylight)
  650. {
  651. mProbeArrayEffect->setCubemapTexture(6, skylightPrefilterMap);
  652. mProbeArrayEffect->setCubemapTexture(7, skylightIrradMap);
  653. }
  654. if (mEffectiveProbeCount != 0)
  655. {
  656. mProbeArrayEffect->setCubemapArrayTexture(4, mPrefilterArray);
  657. mProbeArrayEffect->setCubemapArrayTexture(5, mIrradianceArray);
  658. String useDebugAtten = Con::getVariable("$Probes::showAttenuation", "0");
  659. mProbeArrayEffect->setShaderMacro("DEBUGVIZ_ATTENUATION", useDebugAtten);
  660. String useDebugSpecCubemap = Con::getVariable("$Probes::showSpecularCubemaps", "0");
  661. mProbeArrayEffect->setShaderMacro("DEBUGVIZ_SPECCUBEMAP", useDebugSpecCubemap);
  662. String useDebugDiffuseCubemap = Con::getVariable("$Probes::showDiffuseCubemaps", "0");
  663. mProbeArrayEffect->setShaderMacro("DEBUGVIZ_DIFFCUBEMAP", useDebugDiffuseCubemap);
  664. String useDebugContrib = Con::getVariable("$Probes::showProbeContrib", "0");
  665. mProbeArrayEffect->setShaderMacro("DEBUGVIZ_CONTRIB", useDebugContrib);
  666. if (useDebugContrib == String("1"))
  667. {
  668. MRandomLCG RandomGen;
  669. RandomGen.setSeed(mEffectiveProbeCount);
  670. //also set up some colors
  671. Vector<Point4F> contribColors;
  672. contribColors.setSize(MAXPROBECOUNT);
  673. for (U32 i = 0; i < mEffectiveProbeCount; i++)
  674. {
  675. //we're going to cheat here a little for consistent debugging behavior. The first 3 probes will always have R G and then B for their colors, every other will be random
  676. if (i == 0)
  677. contribColors[i] = Point4F(1, 0, 0, 1);
  678. else if (i == 1)
  679. contribColors[i] = Point4F(0, 1, 0, 1);
  680. else if (i == 2)
  681. contribColors[i] = Point4F(0, 0, 1, 1);
  682. else
  683. contribColors[i] = Point4F(RandomGen.randF(0, 1), RandomGen.randF(0, 1), RandomGen.randF(0, 1),1);
  684. }
  685. mProbeArrayEffect->setShaderConst("$probeContribColors", contribColors);
  686. }
  687. mProbeArrayEffect->setShaderConst("$cubeMips", (float)mMipCount);
  688. mProbeArrayEffect->setShaderConst("$numProbes", (float)mEffectiveProbeCount);
  689. mProbeArrayEffect->setShaderConst("$inProbePosArray", probePositionsData);
  690. mProbeArrayEffect->setShaderConst("$inRefPosArray", probeRefPositionsData);
  691. mProbeArrayEffect->setShaderConst("$worldToObjArray", probeWorldToObjData);
  692. mProbeArrayEffect->setShaderConst("$bbMinArray", probeBBMinData);
  693. mProbeArrayEffect->setShaderConst("$bbMaxArray", probeBBMaxData);
  694. mProbeArrayEffect->setShaderConst("$probeConfigData", probeConfigData);
  695. }
  696. // Make sure the effect is gonna render.
  697. getProbeArrayEffect()->setSkip(false);
  698. //PROFILE_END();
  699. }
  700. void RenderProbeMgr::bakeProbe(ReflectionProbe *probe)
  701. {
  702. GFXDEBUGEVENT_SCOPE(RenderProbeMgr_Bake, ColorI::WHITE);
  703. bool serverObj = probe->isServerObject();
  704. Con::warnf("RenderProbeMgr::bakeProbe() - Beginning bake!");
  705. U32 startMSTime = Platform::getRealMilliseconds();
  706. String path = Con::getVariable("$pref::ReflectionProbes::CurrentLevelPath", "levels/");
  707. U32 resolution = Con::getIntVariable("$pref::ReflectionProbes::BakeResolution", 64);
  708. U32 prefilterMipLevels = mLog2(F32(resolution));
  709. bool renderWithProbes = Con::getIntVariable("$pref::ReflectionProbes::RenderWithProbes", false);
  710. ReflectionProbe *clientProbe = static_cast<ReflectionProbe*>(probe->getClientObject());
  711. if (clientProbe == nullptr)
  712. return;
  713. String probePrefilterPath = clientProbe->getPrefilterMapPath();
  714. String probeIrradPath = clientProbe->getIrradianceMapPath();
  715. if (clientProbe->mReflectionModeType != ReflectionProbe::DynamicCubemap)
  716. {
  717. //Prep our bake path
  718. if (probePrefilterPath.isEmpty() || probeIrradPath.isEmpty())
  719. {
  720. Con::errorf("RenderProbeMgr::bake() - Unable to bake our captures because probe doesn't have a path set");
  721. return;
  722. }
  723. }
  724. // Save the current transforms so we can restore
  725. // it for child control rendering below.
  726. GFXTransformSaver saver;
  727. bool probeRenderState = RenderProbeMgr::smRenderReflectionProbes;
  728. F32 farPlane = 1000.0f;
  729. ReflectorDesc reflDesc;
  730. reflDesc.texSize = resolution;
  731. reflDesc.farDist = farPlane;
  732. reflDesc.detailAdjust = 1;
  733. reflDesc.objectTypeMask = -1;
  734. CubeReflector cubeRefl;
  735. cubeRefl.registerReflector(probe, &reflDesc);
  736. ReflectParams reflParams;
  737. //need to get the query somehow. Likely do some sort of get function to fetch from the guiTSControl that's active
  738. CameraQuery query; //need to get the last cameraQuery
  739. query.fov = 90; //90 degree slices for each of the 6 sides
  740. query.nearPlane = 0.1f;
  741. query.farPlane = farPlane;
  742. query.headMatrix = MatrixF();
  743. query.cameraMatrix = clientProbe->getTransform();
  744. Frustum culler;
  745. culler.set(false,
  746. query.fov,
  747. (F32)resolution / (F32)resolution,
  748. query.nearPlane,
  749. query.farPlane,
  750. query.cameraMatrix);
  751. S32 stereoTarget = GFX->getCurrentStereoTarget();
  752. Point2I maxRes(2048, 2048); //basically a boundary so we don't go over this and break stuff
  753. reflParams.culler = culler;
  754. reflParams.eyeId = stereoTarget;
  755. reflParams.query = &query;
  756. reflParams.startOfUpdateMs = startMSTime;
  757. reflParams.viewportExtent = maxRes;
  758. if (!renderWithProbes)
  759. RenderProbeMgr::smRenderReflectionProbes = false;
  760. cubeRefl.updateReflection(reflParams);
  761. //Now, save out the maps
  762. //create irridiance cubemap
  763. if (cubeRefl.getCubemap())
  764. {
  765. //Just to ensure we're prepped for the generation
  766. clientProbe->createClientResources();
  767. //Prep it with whatever resolution we've dictated for our bake
  768. if (clientProbe->mUseHDRCaptures)
  769. {
  770. clientProbe->mIrridianceMap->mCubemap->initDynamic(resolution, GFXFormatR16G16B16A16F);
  771. clientProbe->mPrefilterMap->mCubemap->initDynamic(resolution, GFXFormatR16G16B16A16F);
  772. }
  773. else
  774. {
  775. clientProbe->mIrridianceMap->mCubemap->initDynamic(resolution, GFXFormatR8G8B8A8);
  776. clientProbe->mPrefilterMap->mCubemap->initDynamic(resolution, GFXFormatR8G8B8A8);
  777. }
  778. GFXTextureTargetRef renderTarget = GFX->allocRenderToTextureTarget(false);
  779. IBLUtilities::GenerateIrradianceMap(renderTarget, cubeRefl.getCubemap(), clientProbe->mIrridianceMap->mCubemap);
  780. IBLUtilities::GeneratePrefilterMap(renderTarget, cubeRefl.getCubemap(), prefilterMipLevels, clientProbe->mPrefilterMap->mCubemap);
  781. U32 endMSTime = Platform::getRealMilliseconds();
  782. F32 diffTime = F32(endMSTime - startMSTime);
  783. Con::warnf("RenderProbeMgr::bake() - Finished Capture! Took %g milliseconds", diffTime);
  784. Con::warnf("RenderProbeMgr::bake() - Beginning save now!");
  785. IBLUtilities::SaveCubeMap(clientProbe->getIrradianceMapPath(), clientProbe->mIrridianceMap->mCubemap);
  786. IBLUtilities::SaveCubeMap(clientProbe->getPrefilterMapPath(), clientProbe->mPrefilterMap->mCubemap);
  787. }
  788. else
  789. {
  790. Con::errorf("RenderProbeMgr::bake() - Didn't generate a valid scene capture cubemap, unable to generate prefilter and irradiance maps!");
  791. }
  792. if (!renderWithProbes)
  793. RenderProbeMgr::smRenderReflectionProbes = probeRenderState;
  794. cubeRefl.unregisterReflector();
  795. U32 endMSTime = Platform::getRealMilliseconds();
  796. F32 diffTime = F32(endMSTime - startMSTime);
  797. Con::warnf("RenderProbeMgr::bake() - Finished bake! Took %g milliseconds", diffTime);
  798. }
  799. void RenderProbeMgr::bakeProbes()
  800. {
  801. //TODO: make this just find every probe in the current missionGroup and run the bake on it automagically
  802. }
  803. DefineEngineMethod(RenderProbeMgr, bakeProbe, void, (ReflectionProbe* probe), (nullAsType< ReflectionProbe*>()),
  804. "@brief returns true if control object is inside the fog\n\n.")
  805. {
  806. if(probe != nullptr)
  807. object->bakeProbe(probe);
  808. }