blob: 2a0d5d6fa87a86833887ce7e3c9a6e1df71ff6bf [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
reed@google.comac10a2d2010-12-22 21:39:39 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
reed@google.comac10a2d2010-12-22 21:39:39 +00007 */
8
9
10#include "GrGLTexture.h"
bsalomon@google.comaa5b6732011-07-29 15:13:20 +000011
reed@google.comac10a2d2010-12-22 21:39:39 +000012#include "GrGpuGL.h"
13
bsalomon@google.com8fe72472011-03-30 21:26:44 +000014#define GPUGL static_cast<GrGpuGL*>(getGpu())
15
bsalomon@google.com0b77d682011-08-19 13:28:54 +000016#define GL_CALL(X) GR_GL_CALL(GPUGL->glInterface(), X)
17
18const GrGLenum* GrGLTexture::WrapMode2GLWrap(GrGLBinding binding) {
twiz@google.comb65e0cb2011-03-18 20:41:44 +000019 static const GrGLenum mirrorRepeatModes[] = {
20 GR_GL_CLAMP_TO_EDGE,
21 GR_GL_REPEAT,
22 GR_GL_MIRRORED_REPEAT
23 };
24
25 static const GrGLenum repeatModes[] = {
26 GR_GL_CLAMP_TO_EDGE,
27 GR_GL_REPEAT,
28 GR_GL_REPEAT
29 };
30
bsalomon@google.com0b77d682011-08-19 13:28:54 +000031 if (kES1_GrGLBinding == binding) {
twiz@google.comb65e0cb2011-03-18 20:41:44 +000032 return repeatModes; // GL_MIRRORED_REPEAT not supported.
33 } else {
34 return mirrorRepeatModes;
35 }
reed@google.comac10a2d2010-12-22 21:39:39 +000036};
37
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000038void GrGLTexture::init(GrGpuGL* gpu,
39 const Desc& textureDesc,
bsalomon@google.com80d09b92011-11-05 21:21:13 +000040 const GrGLRenderTarget::Desc* rtDesc) {
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000041
42 GrAssert(0 != textureDesc.fTextureID);
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000043
bsalomon@google.com80d09b92011-11-05 21:21:13 +000044 fTexParams.invalidate();
45 fTexParamsTimestamp = GrGpu::kExpiredTimestamp;
bsalomon@google.com0b77d682011-08-19 13:28:54 +000046 fTexIDObj = new GrGLTexID(GPUGL->glInterface(),
47 textureDesc.fTextureID,
bsalomon@google.com5877ffd2011-04-11 17:58:48 +000048 textureDesc.fOwnsID);
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000049 fUploadFormat = textureDesc.fUploadFormat;
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000050 fUploadType = textureDesc.fUploadType;
51 fOrientation = textureDesc.fOrientation;
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000052 fScaleX = GrIntToScalar(textureDesc.fContentWidth) /
53 textureDesc.fAllocWidth;
54 fScaleY = GrIntToScalar(textureDesc.fContentHeight) /
55 textureDesc.fAllocHeight;
reed@google.comac10a2d2010-12-22 21:39:39 +000056
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000057 if (NULL != rtDesc) {
bsalomon@google.com8895a7a2011-02-18 16:09:55 +000058 // we render to the top left
59 GrGLIRect vp;
reed@google.comac10a2d2010-12-22 21:39:39 +000060 vp.fLeft = 0;
bsalomon@google.com8895a7a2011-02-18 16:09:55 +000061 vp.fWidth = textureDesc.fContentWidth;
62 vp.fHeight = textureDesc.fContentHeight;
63 vp.fBottom = textureDesc.fAllocHeight - textureDesc.fContentHeight;
64
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000065 fRenderTarget = new GrGLRenderTarget(gpu, *rtDesc, vp, fTexIDObj, this);
reed@google.comac10a2d2010-12-22 21:39:39 +000066 }
reed@google.comac10a2d2010-12-22 21:39:39 +000067}
68
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000069GrGLTexture::GrGLTexture(GrGpuGL* gpu,
bsalomon@google.com80d09b92011-11-05 21:21:13 +000070 const Desc& textureDesc)
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000071 : INHERITED(gpu,
72 textureDesc.fContentWidth,
73 textureDesc.fContentHeight,
bsalomon@google.com0168afc2011-08-08 13:21:05 +000074 textureDesc.fAllocWidth,
75 textureDesc.fAllocHeight,
bsalomon@google.com64c4fe42011-11-05 14:51:01 +000076 textureDesc.fConfig) {
bsalomon@google.com80d09b92011-11-05 21:21:13 +000077 this->init(gpu, textureDesc, NULL);
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000078}
79
80GrGLTexture::GrGLTexture(GrGpuGL* gpu,
81 const Desc& textureDesc,
bsalomon@google.com80d09b92011-11-05 21:21:13 +000082 const GrGLRenderTarget::Desc& rtDesc)
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000083 : INHERITED(gpu,
84 textureDesc.fContentWidth,
85 textureDesc.fContentHeight,
bsalomon@google.com0168afc2011-08-08 13:21:05 +000086 textureDesc.fAllocWidth,
87 textureDesc.fAllocHeight,
bsalomon@google.com64c4fe42011-11-05 14:51:01 +000088 textureDesc.fConfig) {
bsalomon@google.com80d09b92011-11-05 21:21:13 +000089 this->init(gpu, textureDesc, &rtDesc);
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000090}
91
bsalomon@google.com8fe72472011-03-30 21:26:44 +000092void GrGLTexture::onRelease() {
bsalomon@google.com6dcf4992011-04-05 21:16:14 +000093 INHERITED::onRelease();
Scroggoc29d7cd2011-06-16 13:14:21 +000094 GPUGL->notifyTextureDelete(this);
bsalomon@google.com8fe72472011-03-30 21:26:44 +000095 if (NULL != fTexIDObj) {
bsalomon@google.com8fe72472011-03-30 21:26:44 +000096 fTexIDObj->unref();
97 fTexIDObj = NULL;
bsalomon@google.com8fe72472011-03-30 21:26:44 +000098 }
reed@google.comac10a2d2010-12-22 21:39:39 +000099}
100
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000101void GrGLTexture::onAbandon() {
bsalomon@google.com6dcf4992011-04-05 21:16:14 +0000102 INHERITED::onAbandon();
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000103 if (NULL != fTexIDObj) {
104 fTexIDObj->abandon();
105 }
reed@google.comac10a2d2010-12-22 21:39:39 +0000106}
107
bsalomon@google.com79d2dbe2011-06-13 19:28:02 +0000108void GrGLTexture::uploadTextureData(int x,
109 int y,
110 int width,
111 int height,
junov@google.com4ee7ae52011-06-30 17:30:49 +0000112 const void* srcData,
113 size_t rowBytes) {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000114 GrIRect bounds = GrIRect::MakeWH(this->width(), this->height());
115 GrIRect subrect = GrIRect::MakeXYWH(x,y,width, height);
116 if (!bounds.contains(subrect)) {
117 return;
118 }
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000119 GPUGL->setSpareTextureUnit();
bsalomon@google.com8531c1c2011-01-13 19:52:45 +0000120
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000121 // ES2 glCompressedTexSubImage2D doesn't support any formats
reed@google.comac10a2d2010-12-22 21:39:39 +0000122 // (at least without extensions)
bsalomon@google.comc312bf92011-03-21 21:10:33 +0000123 GrAssert(fUploadFormat != GR_GL_PALETTE8_RGBA8);
reed@google.comac10a2d2010-12-22 21:39:39 +0000124
junov@google.com4ee7ae52011-06-30 17:30:49 +0000125 // in case we need a temporary, trimmed copy of the src pixels
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000126 SkAutoSMalloc<128 * 128> tempStorage;
junov@google.com4ee7ae52011-06-30 17:30:49 +0000127
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000128 size_t bpp = GrBytesPerPixel(this->config());
129 size_t trimRowBytes = width * bpp;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000130 if (!rowBytes) {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000131 rowBytes = trimRowBytes;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000132 }
junov@google.com4ee7ae52011-06-30 17:30:49 +0000133 /*
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000134 * check whether to allocate a temporary buffer for flipping y or
135 * because our srcData has extra bytes past each row. If so, we need
bsalomon@google.com7107fa72011-11-10 14:54:14 +0000136 * to trim those off here, since GL ES may not let us specify
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000137 * GL_UNPACK_ROW_LENGTH.
junov@google.com4ee7ae52011-06-30 17:30:49 +0000138 */
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000139 bool restoreGLRowLength = false;
140 bool flipY = kBottomUp_Orientation == fOrientation;
bsalomon@google.com7107fa72011-11-10 14:54:14 +0000141 if (GPUGL->glCaps().fUnpackRowLengthSupport && !flipY) {
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000142 // can't use this for flipping, only non-neg values allowed. :(
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000143 if (srcData && rowBytes != trimRowBytes) {
144 GrGLint rowLength = static_cast<GrGLint>(rowBytes / bpp);
145 GL_CALL(PixelStorei(GR_GL_UNPACK_ROW_LENGTH, rowLength));
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000146 restoreGLRowLength = true;
junov@google.com4ee7ae52011-06-30 17:30:49 +0000147 }
148 } else {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000149 if (srcData && (trimRowBytes != rowBytes || flipY)) {
junov@google.com4ee7ae52011-06-30 17:30:49 +0000150 // copy the data into our new storage, skipping the trailing bytes
151 size_t trimSize = height * trimRowBytes;
152 const char* src = (const char*)srcData;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000153 if (flipY) {
154 src += (height - 1) * rowBytes;
155 }
bsalomon@google.com7d4679a2011-09-02 22:06:24 +0000156 char* dst = (char*)tempStorage.reset(trimSize);
junov@google.com4ee7ae52011-06-30 17:30:49 +0000157 for (int y = 0; y < height; y++) {
158 memcpy(dst, src, trimRowBytes);
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000159 if (flipY) {
160 src -= rowBytes;
161 } else {
162 src += rowBytes;
163 }
junov@google.com4ee7ae52011-06-30 17:30:49 +0000164 dst += trimRowBytes;
165 }
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000166 // now point srcData to our copied version
167 srcData = tempStorage.get();
junov@google.com4ee7ae52011-06-30 17:30:49 +0000168 }
169 }
170
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000171 if (flipY) {
172 y = this->height() - (y + height);
173 }
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000174 GL_CALL(BindTexture(GR_GL_TEXTURE_2D, fTexIDObj->id()));
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000175 GL_CALL(PixelStorei(GR_GL_UNPACK_ALIGNMENT, static_cast<GrGLint>(bpp)));
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000176 GL_CALL(TexSubImage2D(GR_GL_TEXTURE_2D, 0, x, y, width, height,
177 fUploadFormat, fUploadType, srcData));
reed@google.comac10a2d2010-12-22 21:39:39 +0000178
bsalomon@google.com7107fa72011-11-10 14:54:14 +0000179 if (restoreGLRowLength) {
180 GrAssert(GPUGL->glCaps().fUnpackRowLengthSupport);
181 GL_CALL(PixelStorei(GR_GL_UNPACK_ROW_LENGTH, 0));
junov@google.com4ee7ae52011-06-30 17:30:49 +0000182 }
reed@google.comac10a2d2010-12-22 21:39:39 +0000183}
184
bsalomon@google.comcee661a2011-07-26 12:32:36 +0000185intptr_t GrGLTexture::getTextureHandle() const {
bsalomon@google.com1da07462011-03-10 14:51:57 +0000186 return fTexIDObj->id();
reed@google.comac10a2d2010-12-22 21:39:39 +0000187}
188