blob: 29c69b1358206c4334a8dc2a2929e4ec3fbc2aa0 [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,
40 const GrGLRenderTarget::Desc* rtDesc,
41 const TexParams& initialTexParams) {
42
43 GrAssert(0 != textureDesc.fTextureID);
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000044
45 fTexParams = initialTexParams;
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,
70 const Desc& textureDesc,
71 const TexParams& initialTexParams)
72 : INHERITED(gpu,
73 textureDesc.fContentWidth,
74 textureDesc.fContentHeight,
bsalomon@google.com0168afc2011-08-08 13:21:05 +000075 textureDesc.fAllocWidth,
76 textureDesc.fAllocHeight,
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000077 textureDesc.fFormat) {
78 this->init(gpu, textureDesc, NULL, initialTexParams);
79}
80
81GrGLTexture::GrGLTexture(GrGpuGL* gpu,
82 const Desc& textureDesc,
83 const GrGLRenderTarget::Desc& rtDesc,
84 const TexParams& initialTexParams)
85 : INHERITED(gpu,
86 textureDesc.fContentWidth,
87 textureDesc.fContentHeight,
bsalomon@google.com0168afc2011-08-08 13:21:05 +000088 textureDesc.fAllocWidth,
89 textureDesc.fAllocHeight,
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000090 textureDesc.fFormat) {
91 this->init(gpu, textureDesc, &rtDesc, initialTexParams);
92}
93
bsalomon@google.com8fe72472011-03-30 21:26:44 +000094void GrGLTexture::onRelease() {
bsalomon@google.com6dcf4992011-04-05 21:16:14 +000095 INHERITED::onRelease();
Scroggoc29d7cd2011-06-16 13:14:21 +000096 GPUGL->notifyTextureDelete(this);
bsalomon@google.com8fe72472011-03-30 21:26:44 +000097 if (NULL != fTexIDObj) {
bsalomon@google.com8fe72472011-03-30 21:26:44 +000098 fTexIDObj->unref();
99 fTexIDObj = NULL;
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000100 }
reed@google.comac10a2d2010-12-22 21:39:39 +0000101}
102
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000103void GrGLTexture::onAbandon() {
bsalomon@google.com6dcf4992011-04-05 21:16:14 +0000104 INHERITED::onAbandon();
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000105 if (NULL != fTexIDObj) {
106 fTexIDObj->abandon();
107 }
reed@google.comac10a2d2010-12-22 21:39:39 +0000108}
109
bsalomon@google.com79d2dbe2011-06-13 19:28:02 +0000110void GrGLTexture::uploadTextureData(int x,
111 int y,
112 int width,
113 int height,
junov@google.com4ee7ae52011-06-30 17:30:49 +0000114 const void* srcData,
115 size_t rowBytes) {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000116 GrIRect bounds = GrIRect::MakeWH(this->width(), this->height());
117 GrIRect subrect = GrIRect::MakeXYWH(x,y,width, height);
118 if (!bounds.contains(subrect)) {
119 return;
120 }
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000121 GPUGL->setSpareTextureUnit();
bsalomon@google.com8531c1c2011-01-13 19:52:45 +0000122
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000123 // ES2 glCompressedTexSubImage2D doesn't support any formats
reed@google.comac10a2d2010-12-22 21:39:39 +0000124 // (at least without extensions)
bsalomon@google.comc312bf92011-03-21 21:10:33 +0000125 GrAssert(fUploadFormat != GR_GL_PALETTE8_RGBA8);
reed@google.comac10a2d2010-12-22 21:39:39 +0000126
junov@google.com4ee7ae52011-06-30 17:30:49 +0000127 // in case we need a temporary, trimmed copy of the src pixels
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000128 SkAutoSMalloc<128 * 128> tempStorage;
junov@google.com4ee7ae52011-06-30 17:30:49 +0000129
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000130 size_t bpp = GrBytesPerPixel(this->config());
131 size_t trimRowBytes = width * bpp;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000132 if (!rowBytes) {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000133 rowBytes = trimRowBytes;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000134 }
junov@google.com4ee7ae52011-06-30 17:30:49 +0000135 /*
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000136 * check whether to allocate a temporary buffer for flipping y or
137 * because our srcData has extra bytes past each row. If so, we need
138 * to trim those off here, since GL ES doesn't let us specify
139 * GL_UNPACK_ROW_LENGTH.
junov@google.com4ee7ae52011-06-30 17:30:49 +0000140 */
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000141 bool restoreGLRowLength = false;
142 bool flipY = kBottomUp_Orientation == fOrientation;
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000143 if (kDesktop_GrGLBinding == GPUGL->glBinding() && !flipY) {
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000144 // can't use this for flipping, only non-neg values allowed. :(
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000145 if (srcData && rowBytes != trimRowBytes) {
146 GrGLint rowLength = static_cast<GrGLint>(rowBytes / bpp);
147 GL_CALL(PixelStorei(GR_GL_UNPACK_ROW_LENGTH, rowLength));
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000148 restoreGLRowLength = true;
junov@google.com4ee7ae52011-06-30 17:30:49 +0000149 }
150 } else {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000151 if (srcData && (trimRowBytes != rowBytes || flipY)) {
junov@google.com4ee7ae52011-06-30 17:30:49 +0000152 // copy the data into our new storage, skipping the trailing bytes
153 size_t trimSize = height * trimRowBytes;
154 const char* src = (const char*)srcData;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000155 if (flipY) {
156 src += (height - 1) * rowBytes;
157 }
bsalomon@google.com7d4679a2011-09-02 22:06:24 +0000158 char* dst = (char*)tempStorage.reset(trimSize);
junov@google.com4ee7ae52011-06-30 17:30:49 +0000159 for (int y = 0; y < height; y++) {
160 memcpy(dst, src, trimRowBytes);
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000161 if (flipY) {
162 src -= rowBytes;
163 } else {
164 src += rowBytes;
165 }
junov@google.com4ee7ae52011-06-30 17:30:49 +0000166 dst += trimRowBytes;
167 }
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000168 // now point srcData to our copied version
169 srcData = tempStorage.get();
junov@google.com4ee7ae52011-06-30 17:30:49 +0000170 }
171 }
172
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000173 if (flipY) {
174 y = this->height() - (y + height);
175 }
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000176 GL_CALL(BindTexture(GR_GL_TEXTURE_2D, fTexIDObj->id()));
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000177 GL_CALL(PixelStorei(GR_GL_UNPACK_ALIGNMENT, static_cast<GrGLint>(bpp)));
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000178 GL_CALL(TexSubImage2D(GR_GL_TEXTURE_2D, 0, x, y, width, height,
179 fUploadFormat, fUploadType, srcData));
reed@google.comac10a2d2010-12-22 21:39:39 +0000180
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000181 if (kDesktop_GrGLBinding == GPUGL->glBinding()) {
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000182 if (restoreGLRowLength) {
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000183 GL_CALL(PixelStorei(GR_GL_UNPACK_ROW_LENGTH, 0));
junov@google.com4ee7ae52011-06-30 17:30:49 +0000184 }
185 }
reed@google.comac10a2d2010-12-22 21:39:39 +0000186}
187
bsalomon@google.comcee661a2011-07-26 12:32:36 +0000188intptr_t GrGLTexture::getTextureHandle() const {
bsalomon@google.com1da07462011-03-10 14:51:57 +0000189 return fTexIDObj->id();
reed@google.comac10a2d2010-12-22 21:39:39 +0000190}
191