BsGLPixelBuffer.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. //********************************** Banshee Engine (www.banshee3d.com) **************************************************//
  2. //**************** Copyright (c) 2016 Marko Pintera ([email protected]). All rights reserved. **********************//
  3. #include "BsGLPixelBuffer.h"
  4. #include "BsGLTexture.h"
  5. #include "BsGLSupport.h"
  6. #include "BsGLPixelFormat.h"
  7. #include "Error/BsException.h"
  8. #include "Utility/BsBitwise.h"
  9. #include "BsGLRenderTexture.h"
  10. #include "Profiling/BsRenderStats.h"
  11. namespace bs { namespace ct
  12. {
  13. GLPixelBuffer::GLPixelBuffer(UINT32 inWidth, UINT32 inHeight, UINT32 inDepth, PixelFormat inFormat, GpuBufferUsage usage)
  14. : mUsage(usage), mIsLocked(false), mWidth(inWidth), mHeight(inHeight), mDepth(inDepth), mFormat(inFormat)
  15. , mBuffer(inWidth, inHeight, inDepth, inFormat)
  16. {
  17. mSizeInBytes = mHeight*mWidth*PixelUtil::getNumElemBytes(mFormat);
  18. mCurrentLockOptions = (GpuLockOptions)0;
  19. }
  20. GLPixelBuffer::~GLPixelBuffer()
  21. {
  22. mBuffer.freeInternalBuffer();
  23. }
  24. void GLPixelBuffer::allocateBuffer()
  25. {
  26. if(mBuffer.getData())
  27. return;
  28. mBuffer.allocateInternalBuffer();
  29. // TODO: use PBO if we're HBU_DYNAMIC
  30. }
  31. void GLPixelBuffer::freeBuffer()
  32. {
  33. if(mUsage & GBU_STATIC)
  34. mBuffer.freeInternalBuffer();
  35. }
  36. void* GLPixelBuffer::lock(UINT32 offset, UINT32 length, GpuLockOptions options)
  37. {
  38. assert(!mIsLocked && "Cannot lock this buffer, it is already locked!");
  39. assert(offset == 0 && length == mSizeInBytes && "Cannot lock memory region, most lock box or entire buffer");
  40. PixelVolume volume(0, 0, 0, mWidth, mHeight, mDepth);
  41. const PixelData& lockedData = lock(volume, options);
  42. return lockedData.getData();
  43. }
  44. const PixelData& GLPixelBuffer::lock(const PixelVolume& lockBox, GpuLockOptions options)
  45. {
  46. allocateBuffer();
  47. if (options != GBL_WRITE_ONLY_DISCARD)
  48. {
  49. // Download the old contents of the texture
  50. download(mBuffer);
  51. }
  52. mCurrentLockOptions = options;
  53. mLockedBox = lockBox;
  54. mCurrentLock = mBuffer.getSubVolume(lockBox);
  55. mIsLocked = true;
  56. return mCurrentLock;
  57. }
  58. void GLPixelBuffer::unlock()
  59. {
  60. assert(mIsLocked && "Cannot unlock this buffer, it is not locked!");
  61. if (mCurrentLockOptions != GBL_READ_ONLY)
  62. {
  63. // From buffer to card, only upload if was locked for writing
  64. upload(mCurrentLock, mLockedBox);
  65. }
  66. freeBuffer();
  67. mIsLocked = false;
  68. }
  69. void GLPixelBuffer::upload(const PixelData& data, const PixelVolume& dest)
  70. {
  71. BS_EXCEPT(RenderingAPIException, "Upload not possible for this pixel buffer type");
  72. }
  73. void GLPixelBuffer::download(const PixelData& data)
  74. {
  75. BS_EXCEPT(RenderingAPIException, "Download not possible for this pixel buffer type");
  76. }
  77. void GLPixelBuffer::blitFromTexture(GLTextureBuffer* src)
  78. {
  79. blitFromTexture(src,
  80. PixelVolume(0, 0, 0, src->getWidth(), src->getHeight(), src->getDepth()),
  81. PixelVolume(0, 0, 0, mWidth, mHeight, mDepth)
  82. );
  83. }
  84. void GLPixelBuffer::blitFromTexture(GLTextureBuffer* src, const PixelVolume& srcBox, const PixelVolume& dstBox)
  85. {
  86. BS_EXCEPT(RenderingAPIException, "BlitFromTexture not possible for this pixel buffer type");
  87. }
  88. void GLPixelBuffer::bindToFramebuffer(GLenum attachment, UINT32 zoffset, bool allLayers)
  89. {
  90. BS_EXCEPT(RenderingAPIException, "Framebuffer bind not possible for this pixel buffer type");
  91. }
  92. GLTextureBuffer::GLTextureBuffer(
  93. GLenum target,
  94. GLuint id,
  95. GLint face,
  96. GLint level,
  97. GpuBufferUsage usage,
  98. PixelFormat format,
  99. UINT32 multisampleCount)
  100. : GLPixelBuffer(0, 0, 0, format, usage), mTarget(target), mFaceTarget(0), mTextureID(id), mFace(face)
  101. , mLevel(level), mMultisampleCount(multisampleCount)
  102. {
  103. GLint value = 0;
  104. glBindTexture(mTarget, mTextureID);
  105. BS_CHECK_GL_ERROR();
  106. // Get face identifier
  107. mFaceTarget = mTarget;
  108. if(mTarget == GL_TEXTURE_CUBE_MAP)
  109. mFaceTarget = GL_TEXTURE_CUBE_MAP_POSITIVE_X + (face % 6);
  110. // Get width
  111. glGetTexLevelParameteriv(mFaceTarget, level, GL_TEXTURE_WIDTH, &value);
  112. BS_CHECK_GL_ERROR();
  113. mWidth = value;
  114. // Get height
  115. if(target == GL_TEXTURE_1D)
  116. value = 1; // Height always 1 for 1D textures
  117. else
  118. {
  119. glGetTexLevelParameteriv(mFaceTarget, level, GL_TEXTURE_HEIGHT, &value);
  120. BS_CHECK_GL_ERROR();
  121. }
  122. mHeight = value;
  123. // Get depth
  124. if(target != GL_TEXTURE_3D)
  125. value = 1; // Depth always 1 for non-3D textures
  126. else
  127. {
  128. glGetTexLevelParameteriv(mFaceTarget, level, GL_TEXTURE_DEPTH, &value);
  129. BS_CHECK_GL_ERROR();
  130. }
  131. mDepth = value;
  132. // Default
  133. mSizeInBytes = PixelUtil::getMemorySize(mWidth, mHeight, mDepth, mFormat);
  134. // Set up pixel box
  135. mBuffer = PixelData(mWidth, mHeight, mDepth, mFormat);
  136. }
  137. GLTextureBuffer::~GLTextureBuffer()
  138. { }
  139. void GLTextureBuffer::upload(const PixelData& data, const PixelVolume& dest)
  140. {
  141. if ((mUsage & TU_RENDERTARGET) != 0)
  142. {
  143. LOGERR("Writing to render texture from CPU not supported.");
  144. return;
  145. }
  146. if ((mUsage & TU_DEPTHSTENCIL) != 0)
  147. {
  148. LOGERR("Writing to depth stencil texture from CPU not supported.");
  149. return;
  150. }
  151. glBindTexture(mTarget, mTextureID);
  152. BS_CHECK_GL_ERROR();
  153. if(PixelUtil::isCompressed(data.getFormat()))
  154. {
  155. if (data.getFormat() != mFormat || !data.isConsecutive())
  156. {
  157. LOGERR("Compressed images must be consecutive, in the source format");
  158. return;
  159. }
  160. GLenum format = GLPixelUtil::getGLInternalFormat(mFormat);
  161. switch(mTarget) {
  162. case GL_TEXTURE_1D:
  163. if (dest.left == 0)
  164. {
  165. glCompressedTexImage1D(GL_TEXTURE_1D, mLevel,
  166. format,
  167. dest.getWidth(),
  168. 0,
  169. data.getConsecutiveSize(),
  170. data.getData());
  171. BS_CHECK_GL_ERROR();
  172. }
  173. else
  174. {
  175. glCompressedTexSubImage1D(GL_TEXTURE_1D, mLevel,
  176. dest.left,
  177. dest.getWidth(),
  178. format, data.getConsecutiveSize(),
  179. data.getData());
  180. BS_CHECK_GL_ERROR();
  181. }
  182. break;
  183. case GL_TEXTURE_2D:
  184. case GL_TEXTURE_CUBE_MAP:
  185. if (dest.left == 0 && dest.top == 0)
  186. {
  187. glCompressedTexImage2D(mFaceTarget, mLevel,
  188. format,
  189. dest.getWidth(),
  190. dest.getHeight(),
  191. 0,
  192. data.getConsecutiveSize(),
  193. data.getData());
  194. BS_CHECK_GL_ERROR();
  195. }
  196. else
  197. {
  198. glCompressedTexSubImage2D(mFaceTarget, mLevel,
  199. dest.left, dest.top,
  200. dest.getWidth(), dest.getHeight(),
  201. format, data.getConsecutiveSize(),
  202. data.getData());
  203. BS_CHECK_GL_ERROR();
  204. }
  205. break;
  206. case GL_TEXTURE_3D:
  207. if (dest.left == 0 && dest.top == 0 && dest.front == 0)
  208. {
  209. glCompressedTexImage3D(GL_TEXTURE_3D, mLevel,
  210. format,
  211. dest.getWidth(),
  212. dest.getHeight(),
  213. dest.getDepth(),
  214. 0,
  215. data.getConsecutiveSize(),
  216. data.getData());
  217. BS_CHECK_GL_ERROR();
  218. }
  219. else
  220. {
  221. glCompressedTexSubImage3D(GL_TEXTURE_3D, mLevel,
  222. dest.left, dest.top, dest.front,
  223. dest.getWidth(), dest.getHeight(), dest.getDepth(),
  224. format, data.getConsecutiveSize(),
  225. data.getData());
  226. BS_CHECK_GL_ERROR();
  227. }
  228. break;
  229. }
  230. }
  231. else
  232. {
  233. if (data.getWidth() != data.getRowPitch())
  234. {
  235. glPixelStorei(GL_UNPACK_ROW_LENGTH, data.getRowPitch());
  236. BS_CHECK_GL_ERROR();
  237. }
  238. if (data.getHeight()*data.getWidth() != data.getSlicePitch())
  239. {
  240. glPixelStorei(GL_UNPACK_IMAGE_HEIGHT, (data.getSlicePitch() / data.getWidth()));
  241. BS_CHECK_GL_ERROR();
  242. }
  243. if (data.getLeft() > 0 || data.getTop() > 0 || data.getFront() > 0)
  244. {
  245. glPixelStorei(
  246. GL_UNPACK_SKIP_PIXELS,
  247. data.getLeft() + data.getRowPitch() * data.getTop() + data.getSlicePitch() * data.getFront());
  248. BS_CHECK_GL_ERROR();
  249. }
  250. if ((data.getWidth()*PixelUtil::getNumElemBytes(data.getFormat())) & 3)
  251. {
  252. glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
  253. BS_CHECK_GL_ERROR();
  254. }
  255. switch(mTarget) {
  256. case GL_TEXTURE_1D:
  257. glTexSubImage1D(GL_TEXTURE_1D, mLevel,
  258. dest.left,
  259. dest.getWidth(),
  260. GLPixelUtil::getGLOriginFormat(data.getFormat()), GLPixelUtil::getGLOriginDataType(data.getFormat()),
  261. data.getData());
  262. BS_CHECK_GL_ERROR();
  263. break;
  264. case GL_TEXTURE_2D:
  265. case GL_TEXTURE_CUBE_MAP:
  266. glTexSubImage2D(mFaceTarget, mLevel,
  267. dest.left, dest.top,
  268. dest.getWidth(), dest.getHeight(),
  269. GLPixelUtil::getGLOriginFormat(data.getFormat()), GLPixelUtil::getGLOriginDataType(data.getFormat()),
  270. data.getData());
  271. BS_CHECK_GL_ERROR();
  272. break;
  273. case GL_TEXTURE_3D:
  274. glTexSubImage3D(
  275. GL_TEXTURE_3D, mLevel,
  276. dest.left, dest.top, dest.front,
  277. dest.getWidth(), dest.getHeight(), dest.getDepth(),
  278. GLPixelUtil::getGLOriginFormat(data.getFormat()), GLPixelUtil::getGLOriginDataType(data.getFormat()),
  279. data.getData());
  280. BS_CHECK_GL_ERROR();
  281. break;
  282. }
  283. }
  284. // Restore defaults
  285. glPixelStorei(GL_UNPACK_ROW_LENGTH, 0);
  286. BS_CHECK_GL_ERROR();
  287. glPixelStorei(GL_UNPACK_IMAGE_HEIGHT, 0);
  288. BS_CHECK_GL_ERROR();
  289. glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0);
  290. BS_CHECK_GL_ERROR();
  291. glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
  292. BS_CHECK_GL_ERROR();
  293. BS_INC_RENDER_STAT_CAT(ResWrite, RenderStatObject_Texture);
  294. }
  295. void GLTextureBuffer::download(const PixelData &data)
  296. {
  297. if (data.getWidth() != getWidth() || data.getHeight() != getHeight() || data.getDepth() != getDepth())
  298. {
  299. LOGERR("Only download of entire buffer is supported by OpenGL.");
  300. return;
  301. }
  302. glBindTexture(mTarget, mTextureID);
  303. BS_CHECK_GL_ERROR();
  304. if(PixelUtil::isCompressed(data.getFormat()))
  305. {
  306. if (data.getFormat() != mFormat || !data.isConsecutive())
  307. {
  308. LOGERR("Compressed images must be consecutive, in the source format");
  309. return;
  310. }
  311. // Data must be consecutive and at beginning of buffer as PixelStorei not allowed
  312. // for compressed formate
  313. glGetCompressedTexImage(mFaceTarget, mLevel, data.getData());
  314. BS_CHECK_GL_ERROR();
  315. }
  316. else
  317. {
  318. if (data.getWidth() != data.getRowPitch())
  319. {
  320. glPixelStorei(GL_PACK_ROW_LENGTH, data.getRowPitch());
  321. BS_CHECK_GL_ERROR();
  322. }
  323. if (data.getHeight()*data.getWidth() != data.getSlicePitch())
  324. {
  325. glPixelStorei(GL_PACK_IMAGE_HEIGHT, (data.getSlicePitch() / data.getWidth()));
  326. BS_CHECK_GL_ERROR();
  327. }
  328. if (data.getLeft() > 0 || data.getTop() > 0 || data.getFront() > 0)
  329. {
  330. glPixelStorei(
  331. GL_PACK_SKIP_PIXELS,
  332. data.getLeft() + data.getRowPitch() * data.getTop() + data.getSlicePitch() * data.getFront());
  333. BS_CHECK_GL_ERROR();
  334. }
  335. if ((data.getWidth()*PixelUtil::getNumElemBytes(data.getFormat())) & 3)
  336. {
  337. glPixelStorei(GL_PACK_ALIGNMENT, 1);
  338. BS_CHECK_GL_ERROR();
  339. }
  340. // We can only get the entire texture
  341. glGetTexImage(mFaceTarget, mLevel, GLPixelUtil::getGLOriginFormat(data.getFormat()),
  342. GLPixelUtil::getGLOriginDataType(data.getFormat()), data.getData());
  343. BS_CHECK_GL_ERROR();
  344. // Restore defaults
  345. glPixelStorei(GL_PACK_ROW_LENGTH, 0);
  346. BS_CHECK_GL_ERROR();
  347. glPixelStorei(GL_PACK_IMAGE_HEIGHT, 0);
  348. BS_CHECK_GL_ERROR();
  349. glPixelStorei(GL_PACK_SKIP_PIXELS, 0);
  350. BS_CHECK_GL_ERROR();
  351. glPixelStorei(GL_PACK_ALIGNMENT, 4);
  352. BS_CHECK_GL_ERROR();
  353. }
  354. BS_INC_RENDER_STAT_CAT(ResRead, RenderStatObject_Texture);
  355. }
  356. void GLTextureBuffer::bindToFramebuffer(GLenum attachment, UINT32 zoffset, bool allLayers)
  357. {
  358. if(allLayers)
  359. {
  360. switch (mTarget)
  361. {
  362. case GL_TEXTURE_1D:
  363. glFramebufferTexture1D(GL_FRAMEBUFFER, attachment, mFaceTarget, mTextureID, mLevel);
  364. BS_CHECK_GL_ERROR();
  365. break;
  366. case GL_TEXTURE_2D:
  367. glFramebufferTexture2D(GL_FRAMEBUFFER, attachment, mFaceTarget, mTextureID, mLevel);
  368. BS_CHECK_GL_ERROR();
  369. break;
  370. case GL_TEXTURE_2D_MULTISAMPLE:
  371. glFramebufferTexture2D(GL_FRAMEBUFFER, attachment, mFaceTarget, mTextureID, 0);
  372. BS_CHECK_GL_ERROR();
  373. break;
  374. case GL_TEXTURE_CUBE_MAP:
  375. case GL_TEXTURE_3D:
  376. default: // Texture arrays (binding all layers)
  377. glFramebufferTexture(GL_FRAMEBUFFER, attachment, mTextureID, mLevel);
  378. BS_CHECK_GL_ERROR();
  379. break;
  380. }
  381. }
  382. else
  383. {
  384. switch (mTarget)
  385. {
  386. case GL_TEXTURE_3D:
  387. glFramebufferTexture3D(GL_FRAMEBUFFER, attachment, mFaceTarget, mTextureID, mLevel, zoffset);
  388. BS_CHECK_GL_ERROR();
  389. break;
  390. default: // Texture arrays and cube maps
  391. glFramebufferTextureLayer(GL_FRAMEBUFFER, attachment, mTextureID, mLevel, mFace);
  392. BS_CHECK_GL_ERROR();
  393. break;
  394. }
  395. }
  396. }
  397. void GLTextureBuffer::copyFromFramebuffer(UINT32 zoffset)
  398. {
  399. glBindTexture(mTarget, mTextureID);
  400. BS_CHECK_GL_ERROR();
  401. switch(mTarget)
  402. {
  403. case GL_TEXTURE_1D:
  404. glCopyTexSubImage1D(mFaceTarget, mLevel, 0, 0, 0, mWidth);
  405. BS_CHECK_GL_ERROR();
  406. break;
  407. case GL_TEXTURE_2D:
  408. case GL_TEXTURE_CUBE_MAP:
  409. glCopyTexSubImage2D(mFaceTarget, mLevel, 0, 0, 0, 0, mWidth, mHeight);
  410. BS_CHECK_GL_ERROR();
  411. break;
  412. case GL_TEXTURE_3D:
  413. glCopyTexSubImage3D(mFaceTarget, mLevel, 0, 0, zoffset, 0, 0, mWidth, mHeight);
  414. BS_CHECK_GL_ERROR();
  415. break;
  416. }
  417. }
  418. void GLTextureBuffer::blitFromTexture(GLTextureBuffer* src, const PixelVolume& srcBox, const PixelVolume& dstBox)
  419. {
  420. // If supported, prefer direct image copy. If not supported, or if sample counts don't match, fall back to FB blit
  421. #if BS_OPENGL_4_3 || BS_OPENGLES_3_2
  422. if (src->mMultisampleCount > 1 && mMultisampleCount <= 1) // Resolving MS texture
  423. #endif
  424. {
  425. #if BS_OPENGL_4_3 || BS_OPENGLES_3_2
  426. if (mTarget != GL_TEXTURE_2D || mTarget != GL_TEXTURE_2D_MULTISAMPLE)
  427. BS_EXCEPT(InvalidParametersException, "Non-2D multisampled texture not supported.");
  428. #endif
  429. GLint currentFBO = 0;
  430. glGetIntegerv(GL_FRAMEBUFFER_BINDING, &currentFBO);
  431. BS_CHECK_GL_ERROR();
  432. GLuint readFBO = GLRTTManager::instance().getBlitReadFBO();
  433. GLuint drawFBO = GLRTTManager::instance().getBlitDrawFBO();
  434. // Attach source texture
  435. glBindFramebuffer(GL_FRAMEBUFFER, readFBO);
  436. BS_CHECK_GL_ERROR();
  437. src->bindToFramebuffer(0, 0, true);
  438. // Attach destination texture
  439. glBindFramebuffer(GL_FRAMEBUFFER, drawFBO);
  440. BS_CHECK_GL_ERROR();
  441. bindToFramebuffer(0, 0, true);
  442. // Perform blit
  443. glBindFramebuffer(GL_READ_FRAMEBUFFER, readFBO);
  444. BS_CHECK_GL_ERROR();
  445. glBindFramebuffer(GL_DRAW_FRAMEBUFFER, drawFBO);
  446. BS_CHECK_GL_ERROR();
  447. glReadBuffer(GL_COLOR_ATTACHMENT0);
  448. BS_CHECK_GL_ERROR();
  449. glDrawBuffer(GL_COLOR_ATTACHMENT0);
  450. BS_CHECK_GL_ERROR();
  451. glBlitFramebuffer(srcBox.left, srcBox.top, srcBox.right, srcBox.bottom,
  452. dstBox.left, dstBox.top, dstBox.right, dstBox.bottom, GL_COLOR_BUFFER_BIT, GL_NEAREST);
  453. BS_CHECK_GL_ERROR();
  454. // Restore the previously bound FBO
  455. glBindFramebuffer(GL_FRAMEBUFFER, currentFBO);
  456. BS_CHECK_GL_ERROR();
  457. }
  458. #if BS_OPENGL_4_3 || BS_OPENGLES_3_2
  459. else // Just plain copy
  460. {
  461. if (mMultisampleCount != src->mMultisampleCount)
  462. BS_EXCEPT(InvalidParametersException, "When copying textures their multisample counts must match.");
  463. if (mTarget == GL_TEXTURE_3D) // 3D textures can't have arrays so their Z coordinate is handled differently
  464. {
  465. glCopyImageSubData(src->mTextureID, src->mTarget, src->mLevel, srcBox.left, srcBox.top, srcBox.front,
  466. mTextureID, mTarget, mLevel, dstBox.left, dstBox.top, dstBox.front, srcBox.getWidth(), srcBox.getHeight(), srcBox.getDepth());
  467. BS_CHECK_GL_ERROR();
  468. }
  469. else
  470. {
  471. glCopyImageSubData(src->mTextureID, src->mTarget, src->mLevel, srcBox.left, srcBox.top, src->mFace,
  472. mTextureID, mTarget, mLevel, dstBox.left, dstBox.top, mFace, srcBox.getWidth(), srcBox.getHeight(), 1);
  473. BS_CHECK_GL_ERROR();
  474. }
  475. }
  476. #endif
  477. }
  478. }}