blob: 6e93b6c361e63a699df69b7bb74df7aa902eac1b [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
bsalomon@google.com1dcf5062011-11-14 19:29:53 +000018const GrGLenum* GrGLTexture::WrapMode2GLWrap() {
19 static const GrGLenum repeatModes[] = {
twiz@google.comb65e0cb2011-03-18 20:41:44 +000020 GR_GL_CLAMP_TO_EDGE,
21 GR_GL_REPEAT,
22 GR_GL_MIRRORED_REPEAT
23 };
bsalomon@google.com1dcf5062011-11-14 19:29:53 +000024 return repeatModes;
reed@google.comac10a2d2010-12-22 21:39:39 +000025};
26
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000027void GrGLTexture::init(GrGpuGL* gpu,
28 const Desc& textureDesc,
bsalomon@google.com80d09b92011-11-05 21:21:13 +000029 const GrGLRenderTarget::Desc* rtDesc) {
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000030
31 GrAssert(0 != textureDesc.fTextureID);
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000032
bsalomon@google.com80d09b92011-11-05 21:21:13 +000033 fTexParams.invalidate();
34 fTexParamsTimestamp = GrGpu::kExpiredTimestamp;
bsalomon@google.com0b77d682011-08-19 13:28:54 +000035 fTexIDObj = new GrGLTexID(GPUGL->glInterface(),
36 textureDesc.fTextureID,
bsalomon@google.com5877ffd2011-04-11 17:58:48 +000037 textureDesc.fOwnsID);
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000038 fUploadFormat = textureDesc.fUploadFormat;
bsalomon@google.comc6cf7232011-02-17 16:43:10 +000039 fUploadType = textureDesc.fUploadType;
40 fOrientation = textureDesc.fOrientation;
reed@google.comac10a2d2010-12-22 21:39:39 +000041
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000042 if (NULL != rtDesc) {
bsalomon@google.com8895a7a2011-02-18 16:09:55 +000043 // we render to the top left
44 GrGLIRect vp;
reed@google.comac10a2d2010-12-22 21:39:39 +000045 vp.fLeft = 0;
bsalomon@google.com99621082011-11-15 16:47:16 +000046 vp.fWidth = textureDesc.fWidth;
47 vp.fBottom = 0;
48 vp.fHeight = textureDesc.fHeight;
bsalomon@google.com8895a7a2011-02-18 16:09:55 +000049
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000050 fRenderTarget = new GrGLRenderTarget(gpu, *rtDesc, vp, fTexIDObj, this);
reed@google.comac10a2d2010-12-22 21:39:39 +000051 }
reed@google.comac10a2d2010-12-22 21:39:39 +000052}
53
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000054GrGLTexture::GrGLTexture(GrGpuGL* gpu,
bsalomon@google.com80d09b92011-11-05 21:21:13 +000055 const Desc& textureDesc)
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000056 : INHERITED(gpu,
bsalomon@google.com99621082011-11-15 16:47:16 +000057 textureDesc.fWidth,
58 textureDesc.fHeight,
bsalomon@google.com64c4fe42011-11-05 14:51:01 +000059 textureDesc.fConfig) {
bsalomon@google.com80d09b92011-11-05 21:21:13 +000060 this->init(gpu, textureDesc, NULL);
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000061}
62
63GrGLTexture::GrGLTexture(GrGpuGL* gpu,
64 const Desc& textureDesc,
bsalomon@google.com80d09b92011-11-05 21:21:13 +000065 const GrGLRenderTarget::Desc& rtDesc)
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000066 : INHERITED(gpu,
bsalomon@google.com99621082011-11-15 16:47:16 +000067 textureDesc.fWidth,
68 textureDesc.fHeight,
bsalomon@google.com64c4fe42011-11-05 14:51:01 +000069 textureDesc.fConfig) {
bsalomon@google.com80d09b92011-11-05 21:21:13 +000070 this->init(gpu, textureDesc, &rtDesc);
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000071}
72
bsalomon@google.com8fe72472011-03-30 21:26:44 +000073void GrGLTexture::onRelease() {
bsalomon@google.com6dcf4992011-04-05 21:16:14 +000074 INHERITED::onRelease();
Scroggoc29d7cd2011-06-16 13:14:21 +000075 GPUGL->notifyTextureDelete(this);
bsalomon@google.com8fe72472011-03-30 21:26:44 +000076 if (NULL != fTexIDObj) {
bsalomon@google.com8fe72472011-03-30 21:26:44 +000077 fTexIDObj->unref();
78 fTexIDObj = NULL;
bsalomon@google.com8fe72472011-03-30 21:26:44 +000079 }
reed@google.comac10a2d2010-12-22 21:39:39 +000080}
81
bsalomon@google.com8fe72472011-03-30 21:26:44 +000082void GrGLTexture::onAbandon() {
bsalomon@google.com6dcf4992011-04-05 21:16:14 +000083 INHERITED::onAbandon();
bsalomon@google.com8fe72472011-03-30 21:26:44 +000084 if (NULL != fTexIDObj) {
85 fTexIDObj->abandon();
86 }
reed@google.comac10a2d2010-12-22 21:39:39 +000087}
88
bsalomon@google.com79d2dbe2011-06-13 19:28:02 +000089void GrGLTexture::uploadTextureData(int x,
90 int y,
91 int width,
92 int height,
junov@google.com4ee7ae52011-06-30 17:30:49 +000093 const void* srcData,
94 size_t rowBytes) {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +000095 GrIRect bounds = GrIRect::MakeWH(this->width(), this->height());
96 GrIRect subrect = GrIRect::MakeXYWH(x,y,width, height);
97 if (!bounds.contains(subrect)) {
98 return;
99 }
bsalomon@google.com8fe72472011-03-30 21:26:44 +0000100 GPUGL->setSpareTextureUnit();
bsalomon@google.com8531c1c2011-01-13 19:52:45 +0000101
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000102 // ES2 glCompressedTexSubImage2D doesn't support any formats
reed@google.comac10a2d2010-12-22 21:39:39 +0000103 // (at least without extensions)
bsalomon@google.comc312bf92011-03-21 21:10:33 +0000104 GrAssert(fUploadFormat != GR_GL_PALETTE8_RGBA8);
reed@google.comac10a2d2010-12-22 21:39:39 +0000105
junov@google.com4ee7ae52011-06-30 17:30:49 +0000106 // in case we need a temporary, trimmed copy of the src pixels
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000107 SkAutoSMalloc<128 * 128> tempStorage;
junov@google.com4ee7ae52011-06-30 17:30:49 +0000108
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000109 size_t bpp = GrBytesPerPixel(this->config());
110 size_t trimRowBytes = width * bpp;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000111 if (!rowBytes) {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000112 rowBytes = trimRowBytes;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000113 }
junov@google.com4ee7ae52011-06-30 17:30:49 +0000114 /*
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000115 * check whether to allocate a temporary buffer for flipping y or
116 * because our srcData has extra bytes past each row. If so, we need
bsalomon@google.com7107fa72011-11-10 14:54:14 +0000117 * to trim those off here, since GL ES may not let us specify
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000118 * GL_UNPACK_ROW_LENGTH.
junov@google.com4ee7ae52011-06-30 17:30:49 +0000119 */
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000120 bool restoreGLRowLength = false;
121 bool flipY = kBottomUp_Orientation == fOrientation;
bsalomon@google.com7107fa72011-11-10 14:54:14 +0000122 if (GPUGL->glCaps().fUnpackRowLengthSupport && !flipY) {
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000123 // can't use this for flipping, only non-neg values allowed. :(
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000124 if (srcData && rowBytes != trimRowBytes) {
125 GrGLint rowLength = static_cast<GrGLint>(rowBytes / bpp);
126 GL_CALL(PixelStorei(GR_GL_UNPACK_ROW_LENGTH, rowLength));
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000127 restoreGLRowLength = true;
junov@google.com4ee7ae52011-06-30 17:30:49 +0000128 }
129 } else {
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000130 if (srcData && (trimRowBytes != rowBytes || flipY)) {
junov@google.com4ee7ae52011-06-30 17:30:49 +0000131 // copy the data into our new storage, skipping the trailing bytes
132 size_t trimSize = height * trimRowBytes;
133 const char* src = (const char*)srcData;
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000134 if (flipY) {
135 src += (height - 1) * rowBytes;
136 }
bsalomon@google.com7d4679a2011-09-02 22:06:24 +0000137 char* dst = (char*)tempStorage.reset(trimSize);
junov@google.com4ee7ae52011-06-30 17:30:49 +0000138 for (int y = 0; y < height; y++) {
139 memcpy(dst, src, trimRowBytes);
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000140 if (flipY) {
141 src -= rowBytes;
142 } else {
143 src += rowBytes;
144 }
junov@google.com4ee7ae52011-06-30 17:30:49 +0000145 dst += trimRowBytes;
146 }
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000147 // now point srcData to our copied version
148 srcData = tempStorage.get();
junov@google.com4ee7ae52011-06-30 17:30:49 +0000149 }
150 }
151
bsalomon@google.comd9f826c2011-07-18 15:25:04 +0000152 if (flipY) {
153 y = this->height() - (y + height);
154 }
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000155 GL_CALL(BindTexture(GR_GL_TEXTURE_2D, fTexIDObj->id()));
bsalomon@google.com9d6cfd82011-11-05 13:25:21 +0000156 GL_CALL(PixelStorei(GR_GL_UNPACK_ALIGNMENT, static_cast<GrGLint>(bpp)));
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000157 GL_CALL(TexSubImage2D(GR_GL_TEXTURE_2D, 0, x, y, width, height,
158 fUploadFormat, fUploadType, srcData));
reed@google.comac10a2d2010-12-22 21:39:39 +0000159
bsalomon@google.com7107fa72011-11-10 14:54:14 +0000160 if (restoreGLRowLength) {
161 GrAssert(GPUGL->glCaps().fUnpackRowLengthSupport);
162 GL_CALL(PixelStorei(GR_GL_UNPACK_ROW_LENGTH, 0));
junov@google.com4ee7ae52011-06-30 17:30:49 +0000163 }
reed@google.comac10a2d2010-12-22 21:39:39 +0000164}
165
bsalomon@google.comcee661a2011-07-26 12:32:36 +0000166intptr_t GrGLTexture::getTextureHandle() const {
bsalomon@google.com1da07462011-03-10 14:51:57 +0000167 return fTexIDObj->id();
reed@google.comac10a2d2010-12-22 21:39:39 +0000168}
169