mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
7f8ba9c1d7
- Bug 1236786 - [WebGL2] pass getVertexAttrib in gl-object-get-calls.html, r=jgilbert (60a2c91a38)
- Bug 1233046 - Fix OES_texture_float on OSX. - r=jrmuizel (4bc0059f5f)
- Bug 1233557 - Allow RGB8 to be renderable again for web-compat. - r=jrmuizel (4c13bfd8e8)
- Bug 1233549. Disallow ES3 compressed texture formats. r=jgilbert (1073033161)
- Bug 1241702 - Allow unsized DEPTH_STENCIL for RBs in WebGL 2. - r=kamidphish (87d17d2cf9)
- Bug 1239126. Handle gl_InstanceID attribute with no location. r=jgilbert (4894997e98)
- Bug 1236782 - [WebGL2] pass getProgramParameter in gl-object-get-calls.html; r=jgilbert (2136fcce48)
- Bug 1232462. Only ask for a higher version of GLSL when using WebGL2. r=jgilbert (0317be4eb4)
- Bug 1242330 - "Four extensions were promoted to core in WebGL 2 and should no longer be available as extensions." r=jgilbert r=jmuizelaar (6df020b8d4)
- Bug 1233626 - Default MaxDrawingBuffers to 1 unless ext/webgl2. - r=jrmuizel (a7580d661c)
- Bug 1231657. Don't allow linking different versions shaders. r=jgilbert (e610f98066)
- Bug 1241777 - TexCompareFunc should be stored in ascending order. r=jgilbert (b6151a0076)
- Bug 1228885 - Implement WebGLTexture::MemoryUsage. - r=kamidphish (ea06815414)
- Bug 1239259 - Fix WebGL2 generateMipmap checking. r=jgilbert (39f587c421)
- Bug 1242347 - Allow unsized internal format when generate mipmap. r=jgilbert (b203a8898c)
- Bug 1232502. Use the correct internalFormat when calling CopyTexImage2D. r=jgilbert (eeaef3215e)
- Bug 1243663 - Max uniform and attribute location lengths in WebGL2 should be 1024. r=jgilbert (c4ec6de507)
- Bug 1239488 - Add int/uint to vertex attrib data type. r=jgilbert (11b4968025)
- Bug 1184242 - Remove aTabParent != sActiveTabParent warning from IMEStateManager::SetInputContextForChildProcess. r=masayuki (0fcda10e15)
- Bug 1178652 - Send NOTIFY_IME_OF_COMPOSITION_UPDATE to parent process correctly. r=masayuki (bce28e2c91)
- Bug 1107782 - Only accept certain mouse, gamepad events as user-active. r=smaug (00542c80b9)
- Bug 1247850 - Shrink NameTableKey in nsStaticCaseInsensitiveNameTable. r=froydnj,erahm. (ce3cb3edfb)
- Bug 1247359 - micro-optimize the common case of String{Begins,End}With; r=erahm (333e042b31)
- Bug 1239125. Add operator!=(char_type*) to nsTSubstring. r=froydnj (0cc047a9a1)
- Bug 1213862 - Align nsString whitespace handling with web specs; r=froydnj (db5b11ca52)
- Bug 1141884 - Trigger compositor smooth scrolling to snap points when APZ is enabled. r=mstange,kip (593af59f2a)
- Bug 1244582: Add back in a null check that was accidentally removed. r=smaug (76bff1b01f)
- Bug 1234176 - Introduce and use the WriteSysFile() helper function. r=dhylands (22a46fbe8b)
- missing bit of Bug 1198124 - Enable -Wshadow (f84535a7a2)
- Bug 1249171 - Simplify nsCOMArray::SizeOfExcludingThis(). r=erahm. (57efdce1c6)
- Bug 1156416 - Validate camera parameters supplied by the application. r=mikeh (f8b4b84ccf)
- Bug 1186808 - Replace nsBaseHashtable::EnumerateRead() calls in dom/camera/ with iterators. r=mikeh. (7b1db5f6a1)
- Bug 1158378 - Fix how a failed set configuration call would try to shutdown the camera after release. (9d5e323bca)
- Bug 1171374 - Permit software video codecs with the emulated camera. r=sotaro (c1ae26ea0d)
- Bug 1234458 P1 Allow the CacheChild to be "locked" into memory so it will delay destruction. r=ehsan a=ritu (9e46185779)
- Bug 1234458 P2 Lock the CacheChild actor while Cache DOM methods are running. r=ehsan a=ritu (038342a6e2)
- Bug 1244764 P1 Make Cache .add()/.addAll() fail if a Response.ok() is false. r=ehsan (ae26ca9ef1)
- Bug 1172562 - Clear QuotaManager storage when uninstalling an app. Test. r=bkelly (b07311a3b7)
- Bug 1172629 - Use the caches global property from an iframe loaded after setting the pref in order to make the tests pass with the pref disabled; r=bkelly a=RyanVM (e7c05d8b79)
- Bug 1244764 P2 Make dom/cache mochitests pass with new add()/addAll() behavior. r=ehsan (e1f667c1b4)
- Bug 1244764 P3 Make service worker tests pass with new Cache add()/addAll() behavior. r=ehsan (1518ae5225)
- Bug 1003860 - Simplify storage setup tasks in storage inspector tests. r=mratcliffe (249a8bdb2b)
- Bug 1003860 - Service worker cache for storage actor. r=mratcliffe (5c3d1ecd0c)
- Bug 1244764 P5 Fix devtools test to work with new Cache add()/addAll() behavior. r=ehsan (bf85405de8)
- Bug 1232901 - Use channel.asyncOpen2 within dom/browser-element/BrowserElementParent.js (r=sicking,aus) (2a228ed551)
- Bug 1180330 - http auth prompt shown when opening browser if prompt canceled/dismissed earlier. r=fabrice (ba3666f4bd)
- Bug 1234118 - Delete code for supporting 'do-command' and 'copypaste-docommand'. r=mtseng, r=smaug (b1b575d3c5)
- Bug 1238883 - [TV Browser] It shows "The page cannot be displayed" when user browse some webpages. r=roc (e6d7739dd6)
- Bug 1238440 - FileReader should throw an error when the blob changed size when reading, r=khuey (b006adba10)
- Bug 1230422 - FileReader should handle nested ReadAs*() calls. r=khuey (5a3ff84a31)
- Bug 1225202, part 3 - Create files in test_fileapi_slice.html using SpecialPowers.createFiles. r=baku (1137975548)
- Bug 1241171 - FormData should not force 'blob' as filename, r=smaug (748055f751)
- Bug 1246375 - Restore the previous spec version of FormData, r=smaug (3586af2b88)
- Bug 1237183 - Modify implementation of reading preference. r=seanlin (a132bc7246)
- Bug 801545 - Remove DocumentType.internalSubset, r=bz (ea30c9b5ee)
- Bug 1226440 - Expose a method to get a node's immediate dominator; r=bz,sfink (f77ae44037)
- Bug 825318 - Implement adoptDownload for mozDownloadManager, r=aus, r=sicking (e98cb05210)
- Bug 1237370 - Always log the reason for remote AppRep lookup failures. r=gcp (2c804e68fc)
- Bug 1167493 - Application Reputation: disable remote lookup of zip files on Mac/Linux, r=gcp (517459e064)
- Bug 1195519 - Use channel->ascynOpen2 toolkit/components/downloads/ApplicationReputation.cpp (r=sicking) (2856e5213a)
- Bug 1237856 - Add prefs to honor/ignore Application Reputation verdicts. r=gcp (54ee06264f)
- Bug 1243643 - Deprecate unsafe CPOW usage in contentAreaUtils' saveImage. r=jld (6ae790f1ef)
- Bug 1229224: Add an eslint plugin for importing all browser.js globals for browser-chrome tests. r=miker (9df52a7f3b)
- Bug 1245916: Add additional browser window scripts to eslint globals. r=felipe (92d316ca5e)
- Bug 1246244 - Allow non-CPOW documents to pass through saveImageURL properly. r=jaws,Margaret (c8d4ca241d)
- some missing bits after world fix (c0439eebb0)
- add some missing stuff (ddbd47dc03)
- bissing bit of 1229519 (4e255c3dae)
- Bug 1199662 - Crash ping environment block is broken when any string field contains a quotation mark. Unescape INI fields properly using the library that already exists for the purpose. r=ted (874a999edc)
- Bug 1216150 - Turn on the experimental Intl.DateTimeFormat.prototype.formatToParts in b2g certified apps. r=fabrice (40eeb1a4d4)
- Bug 1216150 - Mini-bustage fix for something I think I unintentionally qref'd into the final patch. r=bustage in a CLOSED TREE (36d9b21a67)
- Bug 1141311 - Add async mode support to GonkNativeWindow on Lollipop Gonk r=pchang (39d9d56326)
- Bug 1146671 - Ensure camera not already released when performing operations. r=dhylands (71b59caa1f)
- Bug 1248737. Improve documentation for WorkerRunnable and associated classes. r=khuey (4ff57790c5)
- Bug 1235629 - Remove dead code in WorkerFeature.h, r=smaug (75a51fcf03)
- Bug 1212333 - WorkerDebuggerManager should live on the main thread;r=khuey (11fdfbbae6)
- Bug 1226443 P3 Re-enable service worker update wpt tests. r=ehsan (605dac5f9e)
- Bug 1226443 P4 Cleanup ServiceWorkerScriptCache objects when initialization fails. r=ehsan (43de3429a2)
- Bug 1234127: Change |BluetoothAdapter.pairingReqs| as a nullable object; r=btian, r=mrbkap (45d2038f6a)
- Bug 1188487 - BrowserElement webidl changes for muting and setting volume. r=ehsan (21bea70a07)
- Bug 1238210 - Correct the Promise return types on two Clients methods; r=baku (fa41b25df0)
- Bug 1246784 - Expose Console to the WorkerDebuggerGlobalScope - part 2, r=khuey (0da9ce8ff6)
- Bug 1228702. Don't expose the 'location' property of Exception/DOMException on workers. r=bholley (0fe86ea586)
- Bug 1223825 - Change Directory.path to include the directory's name. r=baku (0cdae4c2f0)
- Bug 1238225 - Mark ExtendableMessageEvent.ports as SameObject; r=baku (45b9a9746f)
- Bug 1236933 - Return null from FetchEvent.clientId for non-subresource network requests; r=bkelly (4a9c4b40cb)
- Bug 1238213 - Make FetchEvent.request non-nullable; r=baku (751082c8ba)
- Bug 1193125 - Avoid corrupting image data in test_fetch_event.html. r=bkelly (9f6bff232f)
- Bug 1201664 - Avoid using Request's constructor when creating FetchEvent.request; r=bkelly (7a3401e345)
- Bug 1175944 - Packaged app's (app://) JS files are not loaded and do not trigger "onfetch" handler. r=jdm (62df139153)
- Bug 1233644 - use pattern matching when listening clear-origin-data. r=baku (ea2594f50e)
- Bug 1237363 - Part 1: Unregister all service workers registered in mochitests at the end of the test; r=jdm (5be97e5bb0)
- Bug 1237363 - Part 2: Fail mochitests which register a service worker without unregistering it; r=jdm (c4160ffd5f)
- Bug 1237363 - Part 3: Add a test for a mochitest finishing without unregistering its service worker; r=jdm (911d37291b)
- Bug 1174078 - Calling "fetch" inside Service Worker's "onfetch" handler in b2g causes "onfetch" again that leads to an infinite loop. Test. r=nsm (208451f346)
- Bug 1197379 - Remove support for intercepting app:// URIs using service workers; r=jdm (3cbdd725f1)
- Bug 1179399 - Part 1: Relax the ShouldIntercept checks when overriding JAR channel info; r=jdm (850bb2bdb8)
- Bug 1238213 follow-up: Mark the FetchEventInit dictionary argument to FetchEvent's constructor optional too; r=bzbarsky (356cbe6db7)
- Bug 1232732 - modify NS_WARNING in MOZ_WIN_MEM_TRY_CATCH; r=aklotz (e2be4d6919)
- Bug 1247658 - Expose a method to JS for find the shortest retaining paths of some nodes in a heap snapshot; r=bz r=jimb (2c82198808)
- Bug 1188115: Expose IDBCursorWithValue in workers. r=baku (e1c40aeb6e)
- Bug 1162680 - Notify Keyboard.jsm to send blur event when the message manager is closed first. r=timdream (53727ab300)
- Bug 1192986 Also mark Cache/CacheStorage as release interfaces on workers. r=ehsan a=bustage (25cf83c154)
- Bug 1159742. Get rid of the pref annotation from test_interfaces, since it basically corresponds to disabling the test. r=jst (c229e3f881)
- Bug 1203160 - Part 2: Fix the interfaces tests to allow SW interfaces for non-release Fennec; r=baku (072840db1f)
- Bug 1197700 - Correct mistakes in InputMethod.webidl. r=kanru, r=janjongboom, sr=smaug (4edb6f201f)
- Bug 1206970 - Stop expecting AnimationPlaybackEvent to be exposed on release branches, where it's disabled by pref, r=smaug (30ae2b13db)
- Bug 1177276 - Pref on canvas.captureStream by default. r=smaug,mt (0cfe0f72f2)
- Bug 1215147 - Enable VR API's on FF for Android by default. r=snorp, r=vlad, r=bz (5ff3725318)
- Bug 1218482 - Enable WebVR By Default,r=bz (f26111ed82)
- Bug 1159755. Stop forcing the media.eme.apiVisible preference to be true in our test harness. r=cpearce (09f7887917)
- Bug 1149312 - Obtain test coverage for the file-backed case of MediaRecorder. r=roc (bd2e7e40f0)
- Bug 1154559 - Remove flaky timeouts from manifest.js and register SimpleTest.registerCleanupFunction() to report unfinished tests. r=cpearce. (eb68db0fb2)
- Bug 1154564 - Add the ability to notify timeouts to MediaTestManager and remove flaky timeouts from test_playback.html. r=cpearce. (c89b4e58d9)
- Bug 1135170 - Fix up racey test_seek-1.html. rpending=mattwoodrow (b3a7d0dcd6)
- Bug 902686 - Change manifest.js to use SpecialPowers.pushPrefEnv. r=edwin (636b0edc1a)
- Bug 1183502 - give androidVersion a correct value in manifest.js. r=sotaro. (933e9ea712)
- Bug 1235588 - add null check to SimpleTest. r=bechen. (958ede68de)
- misspatch (c8922447ff)
- Bug 1151740 - pass the callback object as-is to SpecialPowers.exactGC(). r=edwin (99ca873bce)
- Bug 1197682 - InputMethodManager#setSupportsSwitchingTypes, r=janjongboom, sr=smaug (e7eb54e491)
- Bug 1201407 - Add input-manage-only events for InputMethod API. r=janjongboom, sr=smaug (776d064bd1)
- Bug 1234459 - Expose full text in the input box to InputMethod API, r=masayuki, sr=smaug (4fa0554356)
- Bug 1198163 - Workaround Mochitest app and assign frame proper permissions, r=kanru (c3bcf8ecc1)
- Bug 990250 - Fold nsIStyleSheet into CSSStyleSheet. r=dbaron (23579cb300)
1062 lines
35 KiB
C++
1062 lines
35 KiB
C++
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "WebGLTexture.h"
|
|
|
|
#include <algorithm>
|
|
#include "GLContext.h"
|
|
#include "mozilla/dom/WebGLRenderingContextBinding.h"
|
|
#include "mozilla/gfx/Logging.h"
|
|
#include "mozilla/MathAlgorithms.h"
|
|
#include "mozilla/Scoped.h"
|
|
#include "mozilla/unused.h"
|
|
#include "ScopedGLHelpers.h"
|
|
#include "WebGLContext.h"
|
|
#include "WebGLContextUtils.h"
|
|
#include "WebGLFramebuffer.h"
|
|
#include "WebGLTexelConversions.h"
|
|
|
|
namespace mozilla {
|
|
|
|
/*static*/ const WebGLTexture::ImageInfo WebGLTexture::ImageInfo::kUndefined;
|
|
|
|
////////////////////////////////////////
|
|
|
|
template <typename T>
|
|
static inline T&
|
|
Mutable(const T& x)
|
|
{
|
|
return const_cast<T&>(x);
|
|
}
|
|
|
|
void
|
|
WebGLTexture::ImageInfo::Clear()
|
|
{
|
|
if (!IsDefined())
|
|
return;
|
|
|
|
OnRespecify();
|
|
|
|
Mutable(mFormat) = LOCAL_GL_NONE;
|
|
Mutable(mWidth) = 0;
|
|
Mutable(mHeight) = 0;
|
|
Mutable(mDepth) = 0;
|
|
|
|
MOZ_ASSERT(!IsDefined());
|
|
}
|
|
|
|
WebGLTexture::ImageInfo&
|
|
WebGLTexture::ImageInfo::operator =(const ImageInfo& a)
|
|
{
|
|
MOZ_ASSERT(a.IsDefined());
|
|
|
|
Mutable(mFormat) = a.mFormat;
|
|
Mutable(mWidth) = a.mWidth;
|
|
Mutable(mHeight) = a.mHeight;
|
|
Mutable(mDepth) = a.mDepth;
|
|
|
|
mIsDataInitialized = a.mIsDataInitialized;
|
|
|
|
// But *don't* transfer mAttachPoints!
|
|
MOZ_ASSERT(a.mAttachPoints.empty());
|
|
OnRespecify();
|
|
|
|
return *this;
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::ImageInfo::IsPowerOfTwo() const
|
|
{
|
|
return mozilla::IsPowerOfTwo(mWidth) &&
|
|
mozilla::IsPowerOfTwo(mHeight) &&
|
|
mozilla::IsPowerOfTwo(mDepth);
|
|
}
|
|
|
|
void
|
|
WebGLTexture::ImageInfo::AddAttachPoint(WebGLFBAttachPoint* attachPoint)
|
|
{
|
|
const auto pair = mAttachPoints.insert(attachPoint);
|
|
DebugOnly<bool> didInsert = pair.second;
|
|
MOZ_ASSERT(didInsert);
|
|
}
|
|
|
|
void
|
|
WebGLTexture::ImageInfo::RemoveAttachPoint(WebGLFBAttachPoint* attachPoint)
|
|
{
|
|
DebugOnly<size_t> numElemsErased = mAttachPoints.erase(attachPoint);
|
|
MOZ_ASSERT_IF(IsDefined(), numElemsErased == 1);
|
|
}
|
|
|
|
void
|
|
WebGLTexture::ImageInfo::OnRespecify() const
|
|
{
|
|
for (auto cur : mAttachPoints) {
|
|
cur->OnBackingStoreRespecified();
|
|
}
|
|
}
|
|
|
|
size_t
|
|
WebGLTexture::ImageInfo::MemoryUsage() const
|
|
{
|
|
if (!IsDefined())
|
|
return 0;
|
|
|
|
const auto bytesPerTexel = mFormat->format->estimatedBytesPerPixel;
|
|
return size_t(mWidth) * size_t(mHeight) * size_t(mDepth) * bytesPerTexel;
|
|
}
|
|
|
|
void
|
|
WebGLTexture::ImageInfo::SetIsDataInitialized(bool isDataInitialized, WebGLTexture* tex)
|
|
{
|
|
MOZ_ASSERT(tex);
|
|
MOZ_ASSERT(this >= &tex->mImageInfoArr[0]);
|
|
MOZ_ASSERT(this < &tex->mImageInfoArr[kMaxLevelCount * kMaxFaceCount]);
|
|
|
|
mIsDataInitialized = isDataInitialized;
|
|
tex->InvalidateResolveCache();
|
|
}
|
|
|
|
////////////////////////////////////////
|
|
|
|
JSObject*
|
|
WebGLTexture::WrapObject(JSContext* cx, JS::Handle<JSObject*> givenProto) {
|
|
return dom::WebGLTextureBinding::Wrap(cx, this, givenProto);
|
|
}
|
|
|
|
WebGLTexture::WebGLTexture(WebGLContext* webgl, GLuint tex)
|
|
: WebGLContextBoundObject(webgl)
|
|
, mGLName(tex)
|
|
, mTarget(LOCAL_GL_NONE)
|
|
, mFaceCount(0)
|
|
, mMinFilter(LOCAL_GL_NEAREST_MIPMAP_LINEAR)
|
|
, mMagFilter(LOCAL_GL_LINEAR)
|
|
, mWrapS(LOCAL_GL_REPEAT)
|
|
, mWrapT(LOCAL_GL_REPEAT)
|
|
, mImmutable(false)
|
|
, mImmutableLevelCount(0)
|
|
, mBaseMipmapLevel(0)
|
|
, mMaxMipmapLevel(1000)
|
|
, mTexCompareMode(LOCAL_GL_NONE)
|
|
, mIsResolved(false)
|
|
, mResolved_Swizzle(nullptr)
|
|
{
|
|
mContext->mTextures.insertBack(this);
|
|
}
|
|
|
|
void
|
|
WebGLTexture::Delete()
|
|
{
|
|
for (auto& cur : mImageInfoArr) {
|
|
cur.Clear();
|
|
}
|
|
|
|
mContext->MakeContextCurrent();
|
|
mContext->gl->fDeleteTextures(1, &mGLName);
|
|
|
|
LinkedListElement<WebGLTexture>::removeFrom(mContext->mTextures);
|
|
}
|
|
|
|
size_t
|
|
WebGLTexture::MemoryUsage() const
|
|
{
|
|
if (IsDeleted())
|
|
return 0;
|
|
|
|
size_t accum = 0;
|
|
for (const auto& cur : mImageInfoArr) {
|
|
accum += cur.MemoryUsage();
|
|
}
|
|
return accum;
|
|
}
|
|
|
|
void
|
|
WebGLTexture::SetImageInfo(ImageInfo* target, const ImageInfo& newInfo)
|
|
{
|
|
*target = newInfo;
|
|
|
|
InvalidateResolveCache();
|
|
}
|
|
|
|
void
|
|
WebGLTexture::SetImageInfosAtLevel(uint32_t level, const ImageInfo& newInfo)
|
|
{
|
|
for (uint8_t i = 0; i < mFaceCount; i++) {
|
|
ImageInfoAtFace(i, level) = newInfo;
|
|
}
|
|
|
|
InvalidateResolveCache();
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::IsMipmapComplete(uint32_t texUnit) const
|
|
{
|
|
MOZ_ASSERT(DoesMinFilterRequireMipmap());
|
|
// GLES 3.0.4, p161
|
|
|
|
const uint32_t maxLevel = MaxEffectiveMipmapLevel(texUnit);
|
|
|
|
// "* `level_base <= level_max`"
|
|
if (mBaseMipmapLevel > maxLevel)
|
|
return false;
|
|
|
|
// Make a copy so we can modify it.
|
|
const ImageInfo& baseImageInfo = BaseImageInfo();
|
|
if (!baseImageInfo.IsDefined())
|
|
return false;
|
|
|
|
// Reference dimensions based on the current level.
|
|
uint32_t refWidth = baseImageInfo.mWidth;
|
|
uint32_t refHeight = baseImageInfo.mHeight;
|
|
uint32_t refDepth = baseImageInfo.mDepth;
|
|
MOZ_ASSERT(refWidth && refHeight && refDepth);
|
|
|
|
for (uint32_t level = mBaseMipmapLevel; level <= maxLevel; level++) {
|
|
// "A cube map texture is mipmap complete if each of the six texture images,
|
|
// considered individually, is mipmap complete."
|
|
|
|
for (uint8_t face = 0; face < mFaceCount; face++) {
|
|
const ImageInfo& cur = ImageInfoAtFace(face, level);
|
|
|
|
// "* The set of mipmap arrays `level_base` through `q` (where `q` is defined
|
|
// the "Mipmapping" discussion of section 3.8.10) were each specified with
|
|
// the same effective internal format."
|
|
|
|
// "* The dimensions of the arrays follow the sequence described in the
|
|
// "Mipmapping" discussion of section 3.8.10."
|
|
|
|
if (cur.mWidth != refWidth ||
|
|
cur.mHeight != refHeight ||
|
|
cur.mDepth != refDepth ||
|
|
cur.mFormat != baseImageInfo.mFormat)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// GLES 3.0.4, p158:
|
|
// "[...] until the last array is reached with dimension 1 x 1 x 1."
|
|
if (refWidth == 1 &&
|
|
refHeight == 1 &&
|
|
refDepth == 1)
|
|
{
|
|
break;
|
|
}
|
|
|
|
refWidth = std::max(uint32_t(1), refWidth / 2);
|
|
refHeight = std::max(uint32_t(1), refHeight / 2);
|
|
refDepth = std::max(uint32_t(1), refDepth / 2);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::IsCubeComplete() const
|
|
{
|
|
// GLES 3.0.4, p161
|
|
// "[...] a cube map texture is cube complete if the following conditions all hold
|
|
// true:
|
|
// * The `level_base` arrays of each of the six texture images making up the cube map
|
|
// have identical, positive, and square dimensions.
|
|
// * The `level_base` arrays were each specified with the same effective internal
|
|
// format."
|
|
|
|
// Note that "cube complete" does not imply "mipmap complete".
|
|
|
|
const ImageInfo& reference = BaseImageInfo();
|
|
if (!reference.IsDefined())
|
|
return false;
|
|
|
|
auto refWidth = reference.mWidth;
|
|
auto refFormat = reference.mFormat;
|
|
|
|
for (uint8_t face = 0; face < mFaceCount; face++) {
|
|
const ImageInfo& cur = ImageInfoAtFace(face, mBaseMipmapLevel);
|
|
if (!cur.IsDefined())
|
|
return false;
|
|
|
|
MOZ_ASSERT(cur.mDepth == 1);
|
|
if (cur.mFormat != refFormat || // Check effective formats.
|
|
cur.mWidth != refWidth || // Check both width and height against refWidth to
|
|
cur.mHeight != refWidth) // to enforce positive and square dimensions.
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::IsComplete(uint32_t texUnit, const char** const out_reason) const
|
|
{
|
|
// Texture completeness is established at GLES 3.0.4, p160-161.
|
|
// "[A] texture is complete unless any of the following conditions hold true:"
|
|
|
|
// "* Any dimension of the `level_base` array is not positive."
|
|
const ImageInfo& baseImageInfo = BaseImageInfo();
|
|
if (!baseImageInfo.IsDefined()) {
|
|
// In case of undefined texture image, we don't print any message because this is
|
|
// a very common and often legitimate case (asynchronous texture loading).
|
|
*out_reason = nullptr;
|
|
return false;
|
|
}
|
|
|
|
if (!baseImageInfo.mWidth || !baseImageInfo.mHeight || !baseImageInfo.mDepth) {
|
|
*out_reason = "The dimensions of `level_base` are not all positive.";
|
|
return false;
|
|
}
|
|
|
|
// "* The texture is a cube map texture, and is not cube complete."
|
|
if (IsCubeMap() && !IsCubeComplete()) {
|
|
*out_reason = "Cubemaps must be \"cube complete\".";
|
|
return false;
|
|
}
|
|
|
|
WebGLSampler* sampler = mContext->mBoundSamplers[texUnit];
|
|
TexMinFilter minFilter = sampler ? sampler->mMinFilter : mMinFilter;
|
|
TexMagFilter magFilter = sampler ? sampler->mMagFilter : mMagFilter;
|
|
|
|
// "* The minification filter requires a mipmap (is neither NEAREST nor LINEAR) and
|
|
// the texture is not mipmap complete."
|
|
const bool requiresMipmap = (minFilter != LOCAL_GL_NEAREST &&
|
|
minFilter != LOCAL_GL_LINEAR);
|
|
if (requiresMipmap && !IsMipmapComplete(texUnit)) {
|
|
*out_reason = "Because the minification filter requires mipmapping, the texture"
|
|
" must be \"mipmap complete\".";
|
|
return false;
|
|
}
|
|
|
|
const bool isMinFilteringNearest = (minFilter == LOCAL_GL_NEAREST ||
|
|
minFilter == LOCAL_GL_NEAREST_MIPMAP_NEAREST);
|
|
const bool isMagFilteringNearest = (magFilter == LOCAL_GL_NEAREST);
|
|
const bool isFilteringNearestOnly = (isMinFilteringNearest && isMagFilteringNearest);
|
|
if (!isFilteringNearestOnly) {
|
|
auto formatUsage = baseImageInfo.mFormat;
|
|
auto format = formatUsage->format;
|
|
|
|
// "* The effective internal format specified for the texture arrays is a sized
|
|
// internal color format that is not texture-filterable, and either the
|
|
// magnification filter is not NEAREST or the minification filter is neither
|
|
// NEAREST nor NEAREST_MIPMAP_NEAREST."
|
|
// Since all (GLES3) unsized color formats are filterable just like their sized
|
|
// equivalents, we don't have to care whether its sized or not.
|
|
if (format->isColorFormat && !formatUsage->isFilterable) {
|
|
*out_reason = "Because minification or magnification filtering is not NEAREST"
|
|
" or NEAREST_MIPMAP_NEAREST, and the texture's format is a"
|
|
" color format, its format must be \"texture-filterable\".";
|
|
return false;
|
|
}
|
|
|
|
// "* The effective internal format specified for the texture arrays is a sized
|
|
// internal depth or depth and stencil format, the value of
|
|
// TEXTURE_COMPARE_MODE is NONE[1], and either the magnification filter is not
|
|
// NEAREST, or the minification filter is neither NEAREST nor
|
|
// NEAREST_MIPMAP_NEAREST."
|
|
// [1]: This sounds suspect, but is explicitly noted in the change log for GLES
|
|
// 3.0.1:
|
|
// "* Clarify that a texture is incomplete if it has a depth component, no
|
|
// shadow comparison, and linear filtering (also Bug 9481)."
|
|
// As of OES_packed_depth_stencil rev #3, the sample code explicitly samples from
|
|
// a DEPTH_STENCIL_OES texture with a min-filter of LINEAR. Therefore we relax
|
|
// this restriction if WEBGL_depth_texture is enabled.
|
|
if (!mContext->IsExtensionEnabled(WebGLExtensionID::WEBGL_depth_texture)) {
|
|
if (format->hasDepth && mTexCompareMode != LOCAL_GL_NONE) {
|
|
*out_reason = "A depth or depth-stencil format with TEXTURE_COMPARE_MODE"
|
|
" of NONE must have minification or magnification filtering"
|
|
" of NEAREST or NEAREST_MIPMAP_NEAREST.";
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Texture completeness is effectively (though not explicitly) amended for GLES2 by
|
|
// the "Texture Access" section under $3.8 "Fragment Shaders". This also applies to
|
|
// vertex shaders, as noted on GLES 2.0.25, p41.
|
|
if (!mContext->IsWebGL2()) {
|
|
// GLES 2.0.25, p87-88:
|
|
// "Calling a sampler from a fragment shader will return (R,G,B,A)=(0,0,0,1) if
|
|
// any of the following conditions are true:"
|
|
|
|
// "* A two-dimensional sampler is called, the minification filter is one that
|
|
// requires a mipmap[...], and the sampler's associated texture object is not
|
|
// complete[.]"
|
|
// (already covered)
|
|
|
|
// "* A two-dimensional sampler is called, the minification filter is not one that
|
|
// requires a mipmap (either NEAREST nor[sic] LINEAR), and either dimension of
|
|
// the level zero array of the associated texture object is not positive."
|
|
// (already covered)
|
|
|
|
// "* A two-dimensional sampler is called, the corresponding texture image is a
|
|
// non-power-of-two image[...], and either the texture wrap mode is not
|
|
// CLAMP_TO_EDGE, or the minification filter is neither NEAREST nor LINEAR."
|
|
|
|
// "* A cube map sampler is called, any of the corresponding texture images are
|
|
// non-power-of-two images, and either the texture wrap mode is not
|
|
// CLAMP_TO_EDGE, or the minification filter is neither NEAREST nor LINEAR."
|
|
if (!baseImageInfo.IsPowerOfTwo()) {
|
|
TexWrap wrapS = sampler ? sampler->mWrapS : mWrapS;
|
|
TexWrap wrapT = sampler ? sampler->mWrapT : mWrapT;
|
|
// "either the texture wrap mode is not CLAMP_TO_EDGE"
|
|
if (wrapS != LOCAL_GL_CLAMP_TO_EDGE ||
|
|
wrapT != LOCAL_GL_CLAMP_TO_EDGE)
|
|
{
|
|
*out_reason = "Non-power-of-two textures must have a wrap mode of"
|
|
" CLAMP_TO_EDGE.";
|
|
return false;
|
|
}
|
|
|
|
// "or the minification filter is neither NEAREST nor LINEAR"
|
|
if (requiresMipmap) {
|
|
*out_reason = "Mipmapping requires power-of-two textures.";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// "* A cube map sampler is called, and either the corresponding cube map texture
|
|
// image is not cube complete, or TEXTURE_MIN_FILTER is one that requires a
|
|
// mipmap and the texture is not mipmap cube complete."
|
|
// (already covered)
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
uint32_t
|
|
WebGLTexture::MaxEffectiveMipmapLevel(uint32_t texUnit) const
|
|
{
|
|
WebGLSampler* sampler = mContext->mBoundSamplers[texUnit];
|
|
TexMinFilter minFilter = sampler ? sampler->mMinFilter : mMinFilter;
|
|
if (minFilter == LOCAL_GL_NEAREST ||
|
|
minFilter == LOCAL_GL_LINEAR)
|
|
{
|
|
// No mips used.
|
|
return mBaseMipmapLevel;
|
|
}
|
|
|
|
const auto& imageInfo = BaseImageInfo();
|
|
MOZ_ASSERT(imageInfo.IsDefined());
|
|
|
|
uint32_t maxLevelBySize = mBaseMipmapLevel + imageInfo.MaxMipmapLevels() - 1;
|
|
return std::min<uint32_t>(maxLevelBySize, mMaxMipmapLevel);
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::GetFakeBlackType(const char* funcName, uint32_t texUnit,
|
|
FakeBlackType* const out_fakeBlack)
|
|
{
|
|
const char* incompleteReason;
|
|
if (!IsComplete(texUnit, &incompleteReason)) {
|
|
if (incompleteReason) {
|
|
mContext->GenerateWarning("%s: Active texture %u for target 0x%04x is"
|
|
" 'incomplete', and will be rendered as"
|
|
" RGBA(0,0,0,1), as per the GLES 2.0.24 $3.8.2: %s",
|
|
funcName, texUnit, mTarget.get(),
|
|
incompleteReason);
|
|
}
|
|
*out_fakeBlack = FakeBlackType::RGBA0001;
|
|
return true;
|
|
}
|
|
|
|
// We may still want FakeBlack as an optimization for uninitialized image data.
|
|
bool hasUninitializedData = false;
|
|
bool hasInitializedData = false;
|
|
|
|
const auto maxLevel = MaxEffectiveMipmapLevel(texUnit);
|
|
MOZ_ASSERT(mBaseMipmapLevel <= maxLevel);
|
|
for (uint32_t level = mBaseMipmapLevel; level <= maxLevel; level++) {
|
|
for (uint8_t face = 0; face < mFaceCount; face++) {
|
|
const auto& cur = ImageInfoAtFace(face, level);
|
|
if (cur.IsDataInitialized())
|
|
hasInitializedData = true;
|
|
else
|
|
hasUninitializedData = true;
|
|
}
|
|
}
|
|
MOZ_ASSERT(hasUninitializedData || hasInitializedData);
|
|
|
|
if (!hasUninitializedData) {
|
|
*out_fakeBlack = FakeBlackType::None;
|
|
return true;
|
|
}
|
|
|
|
if (!hasInitializedData) {
|
|
const auto format = ImageInfoAtFace(0, mBaseMipmapLevel).mFormat->format;
|
|
if (format->isColorFormat) {
|
|
*out_fakeBlack = (format->hasAlpha ? FakeBlackType::RGBA0000
|
|
: FakeBlackType::RGBA0001);
|
|
return true;
|
|
}
|
|
|
|
mContext->GenerateWarning("%s: Active texture %u for target 0x%04x is"
|
|
" uninitialized, and will be (perhaps slowly) cleared"
|
|
" by the implementation.",
|
|
funcName, texUnit, mTarget.get());
|
|
} else {
|
|
mContext->GenerateWarning("%s: Active texture %u for target 0x%04x contains"
|
|
" TexImages with uninitialized data along with"
|
|
" TexImages with initialized data, forcing the"
|
|
" implementation to (slowly) initialize the"
|
|
" uninitialized TexImages.",
|
|
funcName, texUnit, mTarget.get());
|
|
}
|
|
|
|
GLenum baseImageTarget = mTarget.get();
|
|
if (baseImageTarget == LOCAL_GL_TEXTURE_CUBE_MAP)
|
|
baseImageTarget = LOCAL_GL_TEXTURE_CUBE_MAP_POSITIVE_X;
|
|
|
|
for (uint32_t level = mBaseMipmapLevel; level <= maxLevel; level++) {
|
|
for (uint8_t face = 0; face < mFaceCount; face++) {
|
|
TexImageTarget imageTarget = baseImageTarget + face;
|
|
if (!EnsureImageDataInitialized(funcName, imageTarget, level))
|
|
return false; // The world just exploded.
|
|
}
|
|
}
|
|
|
|
*out_fakeBlack = FakeBlackType::None;
|
|
return true;
|
|
}
|
|
|
|
static void
|
|
SetSwizzle(gl::GLContext* gl, TexTarget target, const GLint* swizzle)
|
|
{
|
|
static const GLint kNoSwizzle[4] = { LOCAL_GL_RED, LOCAL_GL_GREEN, LOCAL_GL_BLUE,
|
|
LOCAL_GL_ALPHA };
|
|
if (!swizzle) {
|
|
swizzle = kNoSwizzle;
|
|
} else if (!gl->IsSupported(gl::GLFeature::texture_swizzle)) {
|
|
MOZ_CRASH("Needs swizzle feature to swizzle!");
|
|
}
|
|
|
|
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_SWIZZLE_R, swizzle[0]);
|
|
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_SWIZZLE_G, swizzle[1]);
|
|
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_SWIZZLE_B, swizzle[2]);
|
|
gl->fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_SWIZZLE_A, swizzle[3]);
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::ResolveForDraw(const char* funcName, uint32_t texUnit,
|
|
FakeBlackType* const out_fakeBlack)
|
|
{
|
|
if (!mIsResolved) {
|
|
if (!GetFakeBlackType(funcName, texUnit, &mResolved_FakeBlack))
|
|
return false;
|
|
|
|
// Check which swizzle we should use. Since the texture must be complete at this
|
|
// point, just grab the format off any valid image.
|
|
const GLint* newSwizzle = nullptr;
|
|
if (mResolved_FakeBlack == FakeBlackType::None) {
|
|
const auto& cur = ImageInfoAtFace(0, mBaseMipmapLevel);
|
|
newSwizzle = cur.mFormat->textureSwizzleRGBA;
|
|
}
|
|
|
|
// Only set the swizzle if it changed since last time we did it.
|
|
if (newSwizzle != mResolved_Swizzle) {
|
|
mResolved_Swizzle = newSwizzle;
|
|
|
|
// Set the new swizzle!
|
|
mContext->gl->fActiveTexture(LOCAL_GL_TEXTURE0 + texUnit);
|
|
SetSwizzle(mContext->gl, mTarget, mResolved_Swizzle);
|
|
mContext->gl->fActiveTexture(LOCAL_GL_TEXTURE0 + mContext->mActiveTexture);
|
|
}
|
|
|
|
mIsResolved = true;
|
|
}
|
|
|
|
*out_fakeBlack = mResolved_FakeBlack;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::EnsureImageDataInitialized(const char* funcName, TexImageTarget target,
|
|
uint32_t level)
|
|
{
|
|
auto& imageInfo = ImageInfoAt(target, level);
|
|
MOZ_ASSERT(imageInfo.IsDefined());
|
|
|
|
if (imageInfo.IsDataInitialized())
|
|
return true;
|
|
|
|
return InitializeImageData(funcName, target, level);
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::InitializeImageData(const char* funcName, TexImageTarget target,
|
|
uint32_t level)
|
|
{
|
|
auto& imageInfo = ImageInfoAt(target, level);
|
|
MOZ_ASSERT(imageInfo.IsDefined());
|
|
MOZ_ASSERT(!imageInfo.IsDataInitialized());
|
|
|
|
const bool respecifyTexture = false;
|
|
const auto& usage = imageInfo.mFormat;
|
|
const auto& width = imageInfo.mWidth;
|
|
const auto& height = imageInfo.mHeight;
|
|
const auto& depth = imageInfo.mDepth;
|
|
|
|
if (!ZeroTextureData(mContext, funcName, respecifyTexture, mGLName, target, level,
|
|
usage, 0, 0, 0, width, height, depth))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
imageInfo.SetIsDataInitialized(true, this);
|
|
return true;
|
|
}
|
|
|
|
void
|
|
WebGLTexture::ClampLevelBaseAndMax()
|
|
{
|
|
if (!mImmutable)
|
|
return;
|
|
|
|
// GLES 3.0.4, p158:
|
|
// "For immutable-format textures, `level_base` is clamped to the range
|
|
// `[0, levels-1]`, `level_max` is then clamped to the range `
|
|
// `[level_base, levels-1]`, where `levels` is the parameter passed to
|
|
// TexStorage* for the texture object."
|
|
mBaseMipmapLevel = Clamp<uint32_t>(mBaseMipmapLevel, 0, mImmutableLevelCount - 1);
|
|
mMaxMipmapLevel = Clamp<uint32_t>(mMaxMipmapLevel, mBaseMipmapLevel,
|
|
mImmutableLevelCount - 1);
|
|
}
|
|
|
|
void
|
|
WebGLTexture::PopulateMipChain(uint32_t firstLevel, uint32_t lastLevel)
|
|
{
|
|
const ImageInfo& baseImageInfo = ImageInfoAtFace(0, firstLevel);
|
|
MOZ_ASSERT(baseImageInfo.IsDefined());
|
|
|
|
uint32_t refWidth = baseImageInfo.mWidth;
|
|
uint32_t refHeight = baseImageInfo.mHeight;
|
|
uint32_t refDepth = baseImageInfo.mDepth;
|
|
if (!refWidth || !refHeight || !refDepth)
|
|
return;
|
|
|
|
for (uint32_t level = firstLevel + 1; level <= lastLevel; level++) {
|
|
bool isMinimal = (refWidth == 1 &&
|
|
refHeight == 1);
|
|
if (mTarget == LOCAL_GL_TEXTURE_3D) {
|
|
isMinimal &= (refDepth == 1);
|
|
}
|
|
|
|
// Higher levels are unaffected.
|
|
if (isMinimal)
|
|
break;
|
|
|
|
refWidth = std::max(uint32_t(1), refWidth / 2);
|
|
refHeight = std::max(uint32_t(1), refHeight / 2);
|
|
if (mTarget == LOCAL_GL_TEXTURE_3D) { // But not TEXTURE_2D_ARRAY!
|
|
refDepth = std::max(uint32_t(1), refDepth / 2);
|
|
}
|
|
|
|
const ImageInfo cur(baseImageInfo.mFormat, refWidth, refHeight, refDepth,
|
|
baseImageInfo.IsDataInitialized());
|
|
|
|
SetImageInfosAtLevel(level, cur);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// GL calls
|
|
|
|
bool
|
|
WebGLTexture::BindTexture(TexTarget texTarget)
|
|
{
|
|
// silently ignore a deleted texture
|
|
if (IsDeleted())
|
|
return false;
|
|
|
|
const bool isFirstBinding = !HasEverBeenBound();
|
|
if (!isFirstBinding && mTarget != texTarget) {
|
|
mContext->ErrorInvalidOperation("bindTexture: This texture has already been bound"
|
|
" to a different target.");
|
|
return false;
|
|
}
|
|
|
|
mTarget = texTarget;
|
|
|
|
mContext->gl->fBindTexture(mTarget.get(), mGLName);
|
|
|
|
if (isFirstBinding) {
|
|
mFaceCount = IsCubeMap() ? 6 : 1;
|
|
|
|
gl::GLContext* gl = mContext->gl;
|
|
|
|
// Thanks to the WebKit people for finding this out: GL_TEXTURE_WRAP_R
|
|
// is not present in GLES 2, but is present in GL and it seems as if for
|
|
// cube maps we need to set it to GL_CLAMP_TO_EDGE to get the expected
|
|
// GLES behavior.
|
|
// If we are WebGL 2 though, we'll want to leave it as REPEAT.
|
|
const bool hasWrapR = gl->IsSupported(gl::GLFeature::texture_3D);
|
|
if (IsCubeMap() && hasWrapR && !mContext->IsWebGL2()) {
|
|
gl->fTexParameteri(texTarget.get(), LOCAL_GL_TEXTURE_WRAP_R,
|
|
LOCAL_GL_CLAMP_TO_EDGE);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void
|
|
WebGLTexture::GenerateMipmap(TexTarget texTarget)
|
|
{
|
|
// GLES 3.0.4 p160:
|
|
// "Mipmap generation replaces texel array levels level base + 1 through q with arrays
|
|
// derived from the level base array, regardless of their previous contents. All
|
|
// other mipmap arrays, including the level base array, are left unchanged by this
|
|
// computation."
|
|
const ImageInfo& baseImageInfo = BaseImageInfo();
|
|
if (!baseImageInfo.IsDefined()) {
|
|
mContext->ErrorInvalidOperation("generateMipmap: The base level of the texture is"
|
|
" not defined.");
|
|
return;
|
|
}
|
|
|
|
if (IsCubeMap() && !IsCubeComplete()) {
|
|
mContext->ErrorInvalidOperation("generateMipmap: Cube maps must be \"cube"
|
|
" complete\".");
|
|
return;
|
|
}
|
|
|
|
if (!mContext->IsWebGL2() && !baseImageInfo.IsPowerOfTwo()) {
|
|
mContext->ErrorInvalidOperation("generateMipmap: The base level of the texture"
|
|
" does not have power-of-two dimensions.");
|
|
return;
|
|
}
|
|
|
|
auto format = baseImageInfo.mFormat->format;
|
|
if (format->compression) {
|
|
mContext->ErrorInvalidOperation("generateMipmap: Texture data at base level is"
|
|
" compressed.");
|
|
return;
|
|
}
|
|
|
|
if (format->hasDepth) {
|
|
mContext->ErrorInvalidOperation("generateMipmap: Depth textures are not"
|
|
" supported.");
|
|
return;
|
|
}
|
|
|
|
// OpenGL ES 3.0.4 p160:
|
|
// If the level base array was not specified with an unsized internal format from
|
|
// table 3.3 or a sized internal format that is both color-renderable and
|
|
// texture-filterable according to table 3.13, an INVALID_OPERATION error
|
|
// is generated.
|
|
const auto usage = baseImageInfo.mFormat;
|
|
bool canGenerateMipmap = (usage->isRenderable && usage->isFilterable);
|
|
switch (usage->format->effectiveFormat) {
|
|
case webgl::EffectiveFormat::Luminance8:
|
|
case webgl::EffectiveFormat::Alpha8:
|
|
case webgl::EffectiveFormat::Luminance8Alpha8:
|
|
// Non-color-renderable formats from Table 3.3.
|
|
canGenerateMipmap = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (!canGenerateMipmap) {
|
|
mContext->ErrorInvalidOperation("generateMipmap: Texture at base level is not unsized"
|
|
" internal format or is not"
|
|
" color-renderable or texture-filterable.");
|
|
return;
|
|
}
|
|
|
|
// Done with validation. Do the operation.
|
|
|
|
mContext->MakeContextCurrent();
|
|
gl::GLContext* gl = mContext->gl;
|
|
|
|
if (gl->WorkAroundDriverBugs()) {
|
|
// bug 696495 - to work around failures in the texture-mips.html test on various drivers, we
|
|
// set the minification filter before calling glGenerateMipmap. This should not carry a significant performance
|
|
// overhead so we do it unconditionally.
|
|
//
|
|
// note that the choice of GL_NEAREST_MIPMAP_NEAREST really matters. See Chromium bug 101105.
|
|
gl->fTexParameteri(texTarget.get(), LOCAL_GL_TEXTURE_MIN_FILTER,
|
|
LOCAL_GL_NEAREST_MIPMAP_NEAREST);
|
|
gl->fGenerateMipmap(texTarget.get());
|
|
gl->fTexParameteri(texTarget.get(), LOCAL_GL_TEXTURE_MIN_FILTER,
|
|
mMinFilter.get());
|
|
} else {
|
|
gl->fGenerateMipmap(texTarget.get());
|
|
}
|
|
|
|
// Record the results.
|
|
// Note that we don't use MaxEffectiveMipmapLevel() here, since that returns
|
|
// mBaseMipmapLevel if the min filter doesn't require mipmaps.
|
|
const uint32_t lastLevel = mBaseMipmapLevel + baseImageInfo.MaxMipmapLevels() - 1;
|
|
PopulateMipChain(mBaseMipmapLevel, lastLevel);
|
|
}
|
|
|
|
JS::Value
|
|
WebGLTexture::GetTexParameter(TexTarget texTarget, GLenum pname)
|
|
{
|
|
mContext->MakeContextCurrent();
|
|
|
|
GLint i = 0;
|
|
GLfloat f = 0.0f;
|
|
|
|
switch (pname) {
|
|
case LOCAL_GL_TEXTURE_MIN_FILTER:
|
|
case LOCAL_GL_TEXTURE_MAG_FILTER:
|
|
case LOCAL_GL_TEXTURE_WRAP_S:
|
|
case LOCAL_GL_TEXTURE_WRAP_T:
|
|
case LOCAL_GL_TEXTURE_BASE_LEVEL:
|
|
case LOCAL_GL_TEXTURE_COMPARE_FUNC:
|
|
case LOCAL_GL_TEXTURE_COMPARE_MODE:
|
|
case LOCAL_GL_TEXTURE_IMMUTABLE_LEVELS:
|
|
case LOCAL_GL_TEXTURE_MAX_LEVEL:
|
|
case LOCAL_GL_TEXTURE_SWIZZLE_A:
|
|
case LOCAL_GL_TEXTURE_SWIZZLE_B:
|
|
case LOCAL_GL_TEXTURE_SWIZZLE_G:
|
|
case LOCAL_GL_TEXTURE_SWIZZLE_R:
|
|
case LOCAL_GL_TEXTURE_WRAP_R:
|
|
mContext->gl->fGetTexParameteriv(texTarget.get(), pname, &i);
|
|
return JS::NumberValue(uint32_t(i));
|
|
|
|
case LOCAL_GL_TEXTURE_IMMUTABLE_FORMAT:
|
|
mContext->gl->fGetTexParameteriv(texTarget.get(), pname, &i);
|
|
return JS::BooleanValue(bool(i));
|
|
|
|
case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT:
|
|
case LOCAL_GL_TEXTURE_MAX_LOD:
|
|
case LOCAL_GL_TEXTURE_MIN_LOD:
|
|
mContext->gl->fGetTexParameterfv(texTarget.get(), pname, &f);
|
|
return JS::NumberValue(float(f));
|
|
|
|
default:
|
|
MOZ_CRASH("Unhandled pname.");
|
|
}
|
|
}
|
|
|
|
bool
|
|
WebGLTexture::IsTexture() const
|
|
{
|
|
return HasEverBeenBound() && !IsDeleted();
|
|
}
|
|
|
|
// Here we have to support all pnames with both int and float params.
|
|
// See this discussion:
|
|
// https://www.khronos.org/webgl/public-mailing-list/archives/1008/msg00014.html
|
|
void
|
|
WebGLTexture::TexParameter(TexTarget texTarget, GLenum pname, GLint* maybeIntParam,
|
|
GLfloat* maybeFloatParam)
|
|
{
|
|
MOZ_ASSERT(maybeIntParam || maybeFloatParam);
|
|
|
|
GLint intParam = maybeIntParam ? *maybeIntParam : GLint(*maybeFloatParam);
|
|
GLfloat floatParam = maybeFloatParam ? *maybeFloatParam : GLfloat(*maybeIntParam);
|
|
|
|
bool isPNameValid = false;
|
|
switch (pname) {
|
|
// GLES 2.0.25 p76:
|
|
case LOCAL_GL_TEXTURE_WRAP_S:
|
|
case LOCAL_GL_TEXTURE_WRAP_T:
|
|
case LOCAL_GL_TEXTURE_MIN_FILTER:
|
|
case LOCAL_GL_TEXTURE_MAG_FILTER:
|
|
isPNameValid = true;
|
|
break;
|
|
|
|
// GLES 3.0.4 p149-150:
|
|
case LOCAL_GL_TEXTURE_BASE_LEVEL:
|
|
case LOCAL_GL_TEXTURE_COMPARE_MODE:
|
|
case LOCAL_GL_TEXTURE_COMPARE_FUNC:
|
|
case LOCAL_GL_TEXTURE_MAX_LEVEL:
|
|
case LOCAL_GL_TEXTURE_MAX_LOD:
|
|
case LOCAL_GL_TEXTURE_MIN_LOD:
|
|
case LOCAL_GL_TEXTURE_WRAP_R:
|
|
if (mContext->IsWebGL2())
|
|
isPNameValid = true;
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT:
|
|
if (mContext->IsExtensionEnabled(WebGLExtensionID::EXT_texture_filter_anisotropic))
|
|
isPNameValid = true;
|
|
break;
|
|
}
|
|
|
|
if (!isPNameValid) {
|
|
mContext->ErrorInvalidEnumInfo("texParameter: pname", pname);
|
|
return;
|
|
}
|
|
|
|
////////////////
|
|
// Validate params and invalidate if needed.
|
|
|
|
bool paramBadEnum = false;
|
|
bool paramBadValue = false;
|
|
|
|
switch (pname) {
|
|
case LOCAL_GL_TEXTURE_BASE_LEVEL:
|
|
case LOCAL_GL_TEXTURE_MAX_LEVEL:
|
|
paramBadValue = (intParam < 0);
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_COMPARE_MODE:
|
|
paramBadValue = (intParam != LOCAL_GL_NONE &&
|
|
intParam != LOCAL_GL_COMPARE_REF_TO_TEXTURE);
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_COMPARE_FUNC:
|
|
switch (intParam) {
|
|
case LOCAL_GL_LEQUAL:
|
|
case LOCAL_GL_GEQUAL:
|
|
case LOCAL_GL_LESS:
|
|
case LOCAL_GL_GREATER:
|
|
case LOCAL_GL_EQUAL:
|
|
case LOCAL_GL_NOTEQUAL:
|
|
case LOCAL_GL_ALWAYS:
|
|
case LOCAL_GL_NEVER:
|
|
break;
|
|
|
|
default:
|
|
paramBadValue = true;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MIN_FILTER:
|
|
switch (intParam) {
|
|
case LOCAL_GL_NEAREST:
|
|
case LOCAL_GL_LINEAR:
|
|
case LOCAL_GL_NEAREST_MIPMAP_NEAREST:
|
|
case LOCAL_GL_LINEAR_MIPMAP_NEAREST:
|
|
case LOCAL_GL_NEAREST_MIPMAP_LINEAR:
|
|
case LOCAL_GL_LINEAR_MIPMAP_LINEAR:
|
|
break;
|
|
|
|
default:
|
|
paramBadEnum = true;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MAG_FILTER:
|
|
switch (intParam) {
|
|
case LOCAL_GL_NEAREST:
|
|
case LOCAL_GL_LINEAR:
|
|
break;
|
|
|
|
default:
|
|
paramBadEnum = true;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_WRAP_S:
|
|
case LOCAL_GL_TEXTURE_WRAP_T:
|
|
case LOCAL_GL_TEXTURE_WRAP_R:
|
|
switch (intParam) {
|
|
case LOCAL_GL_CLAMP_TO_EDGE:
|
|
case LOCAL_GL_MIRRORED_REPEAT:
|
|
case LOCAL_GL_REPEAT:
|
|
break;
|
|
|
|
default:
|
|
paramBadEnum = true;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT:
|
|
if (maybeFloatParam && floatParam < 1.0f)
|
|
paramBadValue = true;
|
|
else if (maybeIntParam && intParam < 1)
|
|
paramBadValue = true;
|
|
|
|
break;
|
|
}
|
|
|
|
if (paramBadEnum) {
|
|
if (maybeIntParam) {
|
|
mContext->ErrorInvalidEnum("texParameteri: pname 0x%04x: Invalid param"
|
|
" 0x%04x.",
|
|
pname, intParam);
|
|
} else {
|
|
mContext->ErrorInvalidEnum("texParameterf: pname 0x%04x: Invalid param %g.",
|
|
pname, floatParam);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (paramBadValue) {
|
|
if (maybeIntParam) {
|
|
mContext->ErrorInvalidValue("texParameteri: pname 0x%04x: Invalid param %i"
|
|
" (0x%x).",
|
|
pname, intParam, intParam);
|
|
} else {
|
|
mContext->ErrorInvalidValue("texParameterf: pname 0x%04x: Invalid param %g.",
|
|
pname, floatParam);
|
|
}
|
|
return;
|
|
}
|
|
|
|
////////////////
|
|
// Store any needed values
|
|
|
|
switch (pname) {
|
|
case LOCAL_GL_TEXTURE_BASE_LEVEL:
|
|
mBaseMipmapLevel = intParam;
|
|
ClampLevelBaseAndMax();
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MAX_LEVEL:
|
|
mMaxMipmapLevel = intParam;
|
|
ClampLevelBaseAndMax();
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MIN_FILTER:
|
|
mMinFilter = intParam;
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_MAG_FILTER:
|
|
mMagFilter = intParam;
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_WRAP_S:
|
|
mWrapS = intParam;
|
|
break;
|
|
|
|
case LOCAL_GL_TEXTURE_WRAP_T:
|
|
mWrapT = intParam;
|
|
break;
|
|
|
|
// We don't actually need to store the WRAP_R, since it doesn't change texture
|
|
// completeness rules.
|
|
}
|
|
|
|
// Only a couple of pnames don't need to invalidate our resolve status cache.
|
|
switch (pname) {
|
|
case LOCAL_GL_TEXTURE_MAX_ANISOTROPY_EXT:
|
|
case LOCAL_GL_TEXTURE_WRAP_R:
|
|
break;
|
|
|
|
default:
|
|
InvalidateResolveCache();
|
|
break;
|
|
}
|
|
|
|
////////////////
|
|
|
|
mContext->MakeContextCurrent();
|
|
if (maybeIntParam)
|
|
mContext->gl->fTexParameteri(texTarget.get(), pname, intParam);
|
|
else
|
|
mContext->gl->fTexParameterf(texTarget.get(), pname, floatParam);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_0(WebGLTexture)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(WebGLTexture, AddRef)
|
|
NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(WebGLTexture, Release)
|
|
|
|
} // namespace mozilla
|