renderTexTargetBinManager.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  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 "platform/platform.h"
  23. #include "renderInstance/renderTexTargetBinManager.h"
  24. #include "shaderGen/conditionerFeature.h"
  25. #include "core/util/safeDelete.h"
  26. #include "gfx/gfxTextureManager.h"
  27. #include "gfx/gfxStringEnumTranslate.h"
  28. #include "scene/sceneRenderState.h"
  29. #include "scene/sceneManager.h"
  30. IMPLEMENT_CONOBJECT(RenderTexTargetBinManager);
  31. ConsoleDocClass( RenderTexTargetBinManager,
  32. "@brief An abstract base class for render bin managers that render to a named textue target.\n\n"
  33. "This bin itself doesn't do any rendering work. It offers functionality to manage "
  34. "a texture render target which derived render bin classes can render into.\n\n"
  35. "@see RenderDeferredMgr\n"
  36. "@ingroup RenderBin\n" );
  37. RenderTexTargetBinManager::RenderTexTargetBinManager( const RenderInstType& ritype,
  38. F32 renderOrder,
  39. F32 processAddOrder,
  40. const GFXFormat targetFormat,
  41. const Point2I &targetSize,
  42. const U32 targetChainLength )
  43. : Parent( ritype, renderOrder, processAddOrder ),
  44. mTargetFormat(targetFormat),
  45. mTargetSize(targetSize),
  46. mTargetScale(1.0f, 1.0f),
  47. mTargetSizeType(FixedSize),
  48. mTargetChainLength(targetChainLength),
  49. mTargetChainIdx(0),
  50. mNumRenderTargets(1),
  51. mTargetChain(NULL),
  52. mTargetChainTextures(NULL)
  53. #ifndef TORQUE_SHIPPING
  54. ,m_NeedsOnPostRender(false)
  55. #endif
  56. {
  57. GFXDevice::getDeviceEventSignal().notify(this, &RenderTexTargetBinManager::_handleGFXEvent);
  58. GFXTextureManager::addEventDelegate( this, &RenderTexTargetBinManager::_onTextureEvent );
  59. mNamedTarget.setSamplerState( GFXSamplerStateDesc::getClampPoint() );
  60. }
  61. RenderTexTargetBinManager::~RenderTexTargetBinManager()
  62. {
  63. _teardownTargets();
  64. GFXTextureManager::removeEventDelegate( this, &RenderTexTargetBinManager::_onTextureEvent );
  65. GFXDevice::getDeviceEventSignal().remove(this, &RenderTexTargetBinManager::_handleGFXEvent);
  66. }
  67. bool RenderTexTargetBinManager::onAdd()
  68. {
  69. if(!Parent::onAdd())
  70. return false;
  71. _setupTargets();
  72. return true;
  73. }
  74. ImplementEnumType( RenderTexTargetSize,
  75. "What size to render the target texture. Sizes are based on the Window the render is occuring in.\n"
  76. "@ingroup RenderBin\n\n")
  77. { RenderTexTargetBinManager::WindowSize, "windowsize", "Render to the size of the window.\n" },
  78. { RenderTexTargetBinManager::WindowSizeScaled, "windowsizescaled", "Render to the size of the window, scaled to the render target's size.\n" },
  79. { RenderTexTargetBinManager::FixedSize, "fixedsize", "Don't scale the target texture, and render to its default size.\n" },
  80. EndImplementEnumType;
  81. void RenderTexTargetBinManager::initPersistFields()
  82. {
  83. // TOM_TODO:
  84. //addField( "targetScale", mTargetScale );
  85. //addPropertyNOPS( "targetSizeType", mTargetSizeType)->setEnumTable(gSizeTypeEnumTable);
  86. //addPropertyNOPS<S32>( "targetFormat")->setEnumTable(gTextureFormatEnumTable)->addGet(this, &RenderTexTargetBinManager::getTargetFormatConsole)->addSet(this, &RenderTexTargetBinManager::setTargetFormatConsole);
  87. //addProperty<bool>( "blur" )->addSet(this, &RenderTexTargetBinManager::setBlur)->addGet(this, &RenderTexTargetBinManager::getBlur);
  88. Parent::initPersistFields();
  89. }
  90. bool RenderTexTargetBinManager::setTargetSize(const Point2I &newTargetSize)
  91. {
  92. if( mTargetSize.x >= newTargetSize.x &&
  93. mTargetSize.y >= newTargetSize.y )
  94. return true;
  95. mTargetSize = newTargetSize;
  96. mNamedTarget.setViewport( RectI( Point2I::Zero, mTargetSize ) );
  97. return _updateTargets();
  98. }
  99. bool RenderTexTargetBinManager::setTargetFormat(const GFXFormat &newTargetFormat)
  100. {
  101. if(mTargetFormat == newTargetFormat)
  102. return true;
  103. mTargetFormat = newTargetFormat;
  104. ConditionerFeature *conditioner = mNamedTarget.getConditioner();
  105. if(conditioner)
  106. conditioner->setBufferFormat(mTargetFormat);
  107. return _updateTargets();
  108. }
  109. void RenderTexTargetBinManager::setTargetChainLength(const U32 chainLength)
  110. {
  111. if(mTargetChainLength != chainLength)
  112. {
  113. mTargetChainLength = chainLength;
  114. _setupTargets();
  115. }
  116. }
  117. GFXTextureObject* RenderTexTargetBinManager::getTargetTexture( U32 mrtIndex, S32 chainIndex ) const
  118. {
  119. const U32 chainIdx = ( chainIndex > -1 ) ? chainIndex : mTargetChainIdx;
  120. if(chainIdx < mTargetChainLength)
  121. return mTargetChainTextures[chainIdx][mrtIndex];
  122. return NULL;
  123. }
  124. bool RenderTexTargetBinManager::_updateTargets()
  125. {
  126. PROFILE_SCOPE(RenderTexTargetBinManager_updateTargets);
  127. bool ret = true;
  128. mNamedTarget.release();
  129. // Update the target size
  130. for( U32 i = 0; i < mTargetChainLength; i++ )
  131. {
  132. AssertFatal( mTargetChain != NULL, "RenderTexTargetBinManager - target chain not set up" );
  133. if( !mTargetChain[i] )
  134. mTargetChain[i] = GFX->allocRenderToTextureTarget();
  135. for( U32 j = 0; j < mNumRenderTargets; j++ )
  136. {
  137. // try reuse of old color texture
  138. if( mTargetChainTextures[i][j].getWidthHeight() != mTargetSize
  139. || mTargetChainTextures[i][j].getFormat() != mTargetFormat)
  140. {
  141. ret &= mTargetChainTextures[i][j].set( mTargetSize.x, mTargetSize.y, mTargetFormat,
  142. &GFXRenderTargetProfile, avar( "%s() - (line %d)", __FUNCTION__, __LINE__ ),
  143. 1, GFXTextureManager::AA_MATCH_BACKBUFFER );
  144. mTargetChain[i]->attachTexture( GFXTextureTarget::RenderSlot(GFXTextureTarget::Color0 + j), mTargetChainTextures[i][j] );
  145. }
  146. }
  147. }
  148. return ret;
  149. }
  150. bool RenderTexTargetBinManager::_handleGFXEvent( GFXDevice::GFXDeviceEventType event_ )
  151. {
  152. if ( event_ == GFXDevice::deStartOfFrame )
  153. {
  154. mTargetChainIdx++;
  155. if ( mTargetChainIdx >= mTargetChainLength )
  156. mTargetChainIdx = 0;
  157. }
  158. return true;
  159. }
  160. void RenderTexTargetBinManager::_onTextureEvent( GFXTexCallbackCode code )
  161. {
  162. switch(code)
  163. {
  164. case GFXZombify:
  165. _teardownTargets();
  166. break;
  167. case GFXResurrect:
  168. _setupTargets();
  169. break;
  170. }
  171. }
  172. bool RenderTexTargetBinManager::_setupTargets()
  173. {
  174. _teardownTargets();
  175. mTargetChain = new GFXTextureTargetRef[mTargetChainLength];
  176. mTargetChainTextures = new GFXTexHandle*[mTargetChainLength];
  177. for( U32 i = 0; i < mTargetChainLength; i++ )
  178. mTargetChainTextures[i] = new GFXTexHandle[mNumRenderTargets];
  179. mTargetChainIdx = 0;
  180. mTargetSize = Point2I::Zero;
  181. return true;
  182. }
  183. void RenderTexTargetBinManager::_teardownTargets()
  184. {
  185. mNamedTarget.release();
  186. SAFE_DELETE_ARRAY(mTargetChain);
  187. if(mTargetChainTextures != NULL)
  188. {
  189. for( U32 i = 0; i < mTargetChainLength; i++ )
  190. SAFE_DELETE_ARRAY(mTargetChainTextures[i]);
  191. }
  192. SAFE_DELETE_ARRAY(mTargetChainTextures);
  193. }
  194. GFXTextureTargetRef RenderTexTargetBinManager::_getTextureTarget(const U32 idx /* = 0 */)
  195. {
  196. return mTargetChain[idx];
  197. }
  198. bool RenderTexTargetBinManager::_onPreRender(SceneRenderState * state, bool preserve /* = false */)
  199. {
  200. PROFILE_SCOPE(RenderTexTargetBinManager_onPreRender);
  201. #ifndef TORQUE_SHIPPING
  202. AssertFatal( m_NeedsOnPostRender == false, "_onPostRender not called on RenderTexTargetBinManager, or sub-class." );
  203. m_NeedsOnPostRender = false;
  204. #endif
  205. // Update the render target size
  206. const Point2I &rtSize = GFX->getActiveRenderTarget()->getSize();
  207. switch(mTargetSizeType)
  208. {
  209. case WindowSize:
  210. setTargetSize(rtSize);
  211. break;
  212. case WindowSizeScaled:
  213. {
  214. Point2I scaledTargetSize(mFloor(rtSize.x * mTargetScale.x), mFloor(rtSize.y * mTargetScale.y));
  215. setTargetSize(scaledTargetSize);
  216. break;
  217. }
  218. case FixedSize:
  219. // No adjustment necessary
  220. break;
  221. }
  222. if( mTargetChainLength == 0 )
  223. return false;
  224. GFXTextureTargetRef binTarget = _getTextureTarget(mTargetChainIdx);
  225. if( binTarget.isNull() )
  226. return false;
  227. // Attach active depth target texture
  228. binTarget->attachTexture(GFXTextureTarget::DepthStencil, getRenderPass()->getDepthTargetTexture());
  229. // Preserve contents
  230. if(preserve)
  231. GFX->getActiveRenderTarget()->preserve();
  232. GFX->pushActiveRenderTarget();
  233. GFX->setActiveRenderTarget(binTarget);
  234. GFX->setViewport( mNamedTarget.getViewport() );
  235. #ifndef TORQUE_SHIPPING
  236. m_NeedsOnPostRender = true;
  237. #endif
  238. return true;
  239. }
  240. void RenderTexTargetBinManager::_onPostRender()
  241. {
  242. PROFILE_SCOPE(RenderTexTargetBinManager_onPostRender);
  243. #ifndef TORQUE_SHIPPING
  244. m_NeedsOnPostRender = false;
  245. #endif
  246. GFXTextureTargetRef binTarget = _getTextureTarget(mTargetChainIdx);
  247. binTarget->resolve();
  248. GFX->popActiveRenderTarget();
  249. for ( U32 i=0; i < mNumRenderTargets; i++ )
  250. mNamedTarget.setTexture( i, mTargetChainTextures[mTargetChainIdx][i] );
  251. }