1#include "CapabilitiesGLES30.h"
9#include "CommonGLES30.h"
10#include "FormatsGLES30.h"
12#include "Foundation/Logging/Logger.h"
15#include <emscripten.h>
22 void toUpper(std::string &
string)
24 std::transform(
string.begin(),
string.end(),
string.begin(), [](
const char & c) {
return (
char)::toupper(c); });
28void Cogs::CapabilitiesGLES30::initialize(
bool isFullGL,
bool useClipControl)
32 std::string vendor((
const char *)glGetString(GL_VENDOR));
33 std::string renderer((
const char *)glGetString(GL_RENDERER));
36 LOG_DEBUG(logger,
"GL_VERSION: %s", glGetString(GL_VERSION));
37 LOG_DEBUG(logger,
"GL_VENDOR: %s", vendor.c_str());
38 LOG_DEBUG(logger,
"GL_RENDERER: %s", renderer.c_str());
39 LOG_DEBUG(logger,
"GL_SHADING_LANGUAGE_VERSION: %s", glGetString(GL_SHADING_LANGUAGE_VERSION));
43 const auto extensionsString = std::string((
const char *)glGetString(GL_EXTENSIONS));
45 std::istringstream sstream(extensionsString);
46 extensions = std::unordered_set<std::string>(std::istream_iterator<std::string>(sstream), std::istream_iterator<std::string>());
50 std::istringstream sstream(extensionsString);
52 while(sstream >> ext){
53 LOG_DEBUG(logger,
"Extension: %s", ext.c_str());
60 if (vendor.find(
"NVIDIA") != vendor.npos) {
63 if (renderer.find(
"QUADRO FX") != std::string::npos) series =
Series::QuadroFX;
64 else if (renderer.find(
"QUADRO") != std::string::npos) series =
Series::Quadro;
65 else if (renderer.find(
"GEFORCE") != std::string::npos) series =
Series::geForce;
67 auto modelString = renderer.substr(0, renderer.find_first_of(
'/'));
69 if (modelString.size()) {
70 std::regex numbers(
"\\d.*");
73 if (std::regex_search(modelString, results, numbers)) {
74 model = std::stoi(results[0].str());
77 }
else if (vendor.find(
"AMD") != vendor.npos || vendor.find(
"ATI") != vendor.npos) {
80 if (renderer.find(
"FIREGL") != std::string::npos) series =
Series::FireGL;
81 }
else if (vendor.find(
"INTEL") != vendor.npos) {
87 glGetIntegerv(GL_MAX_TEXTURE_IMAGE_UNITS, &deviceCapabilities.MaxTextureSlots);
88 glGetIntegerv(GL_MAX_VERTEX_ATTRIBS, &deviceCapabilities.MaxVertexInputElements);
90 if (isSupported(
"GL_EXT_texture_filter_anisotropic")) {
91 glGetFloatv(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT, &deviceCapabilities.MaxAnisotropy);
92 LOG_DEBUG(logger,
"GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT: %f", deviceCapabilities.MaxAnisotropy);
96 glGetIntegerv(GL_MAX_SAMPLES, &samples);
97 deviceCapabilities.MaxSamples = std::max(1, samples);
99 GLint maxTextureSize = 0;
100 glGetIntegerv(GL_MAX_TEXTURE_SIZE, &maxTextureSize);
103 GLint max3DTextureSize = 0;
104 glGetIntegerv(GL_MAX_3D_TEXTURE_SIZE, &max3DTextureSize);
107 GLint maxCubeTextureSize = 0;
108 glGetIntegerv(GL_MAX_CUBE_MAP_TEXTURE_SIZE, &maxCubeTextureSize);
111 GLint maxArrayTextureLayers = 0;
112 glGetIntegerv(GL_MAX_ARRAY_TEXTURE_LAYERS, &maxArrayTextureLayers);
115 LOG_DEBUG(logger,
"Texture limits: 2D=%u 3D=%u Cube=%u Array=%u",
121 GLint uniformBufferOffsetAlignment = 0;
122 glGetIntegerv(GL_UNIFORM_BUFFER_OFFSET_ALIGNMENT, &uniformBufferOffsetAlignment);
123 if (uniformBufferOffsetAlignment) {
128 if (isSupported(
"GL_OCULUS_multiview")) {
129 deviceCapabilities.MultiSampleMultiView =
true;
130 deviceCapabilities.MultiView =
true;
132 else if (isSupported(
"GL_OVR_multiview_multisampled_render_to_texture")) {
133 deviceCapabilities.MultiSampleMultiView =
true;
134 deviceCapabilities.MultiView =
true;
136 else if (isSupported(
"GL_OVR_multiview2")) {
137 deviceCapabilities.MultiView =
true;
139 else if (isSupported(
"GL_OVR_multiview")) {
140 deviceCapabilities.MultiView =
true;
144 if (!deviceCapabilities.MultiView) {
146 for (
const item of Object.values(Module.GL.contexts)) {
147 if (item && item.hasOwnProperty(
'GLctx')) {
148 const ext = item.GLctx.getExtension(
"OVR_multiview2");
149 return ext !== null ? 1 : 0;
154 LOG_DEBUG(logger,
"Browser supports OVR_multiview2 even though emscripten doesn't, enabling regardless");
155 deviceCapabilities.MultiView =
true;
159 if (!deviceCapabilities.MultiSampleMultiView) {
161 for (
const item of Object.values(Module.GL.contexts)) {
162 if (item && item.hasOwnProperty(
'GLctx')) {
163 const ext = item.GLctx.getExtension(
"OCULUS_multiview");
164 return ext !== null ? 1 : 0;
169 LOG_DEBUG(logger,
"Browser supports OCULUS_multiview even though emscripten doesn't, enabling regardless");
170 deviceCapabilities.MultiSampleMultiView =
true;
171 deviceCapabilities.MultiView =
true;
177 deviceCapabilities.DefaultColorTargetHasLinearEncoding =
false;
179 GLint param = GL_INVALID_ENUM;
180 glBindFramebuffer(GL_FRAMEBUFFER, 0);
181 glGetFramebufferAttachmentParameteriv(GL_FRAMEBUFFER, GL_BACK, GL_FRAMEBUFFER_ATTACHMENT_COLOR_ENCODING, ¶m);
184 LOG_DEBUG(logger,
"Default render target has LINEAR encoding");
185 deviceCapabilities.DefaultColorTargetHasLinearEncoding =
true;
188 LOG_DEBUG(logger,
"Default render target has SRGB encoding");
191 LOG_ERROR(logger,
"glGetFramebufferAttachmentParameter returned unrecognized enum: #%04x", param);
196 if (deviceCapabilities.MultiView) {
198 glGetIntegerv(GL_MAX_VIEWS_OVR, &maxViews);
199 deviceCapabilities.MaxMultiViews = std::max(0, maxViews);
200 LOG_DEBUG(logger,
"MultiView rendering available, maxViews=%d, maxSamples=%d",
202 deviceCapabilities.MultiSampleMultiView ? deviceCapabilities.MaxSamples : 1);
209 deviceCapabilities.VertexAndIndexDataInSharedBuffers =
false;
212 deviceCapabilities.SyncObjects =
true;
213 deviceCapabilities.UnsignedIntIndexes =
true;
214 deviceCapabilities.FloatTextures =
true;
215 deviceCapabilities.FragDepth =
true;
216 deviceCapabilities.SupportsHlsl =
false;
217 deviceCapabilities.StartInstance =
false;
218 deviceCapabilities.IndependentSamplerState =
false;
219 deviceCapabilities.SupportsMultipleThreads =
false;
220 deviceCapabilities.TextureCubeArrays =
false;
221 deviceCapabilities.VertexArrayObjects =
true;
222 deviceCapabilities.OriginOnTop =
false;
223 deviceCapabilities.DepthNegativeOneToOne = !useClipControl;
225 if (isSupported(
"GL_EXT_depth_clamp")) {
232 if (isSupported(
"GL_EXT_texture_compression_s3tc") ||
233 isSupported(
"GL_WEBGL_compressed_texture_s3tc")) {
234 deviceCapabilities.TextureCompressionS3TC =
true;
236 if (isSupported(
"GL_ARB_texture_compression_rgtc") ||
237 isSupported(
"GL_EXT_texture_compression_rgtc")) {
238 deviceCapabilities.TextureCompressionRGTC =
true;
240 if (isSupported(
"GL_ARB_texture_compression_bptc") ||
241 isSupported(
"GL_EXT_texture_compression_bptc")) {
242 deviceCapabilities.TextureCompressionBPTC =
true;
244 if (isSupported(
"WEBGL_compressed_texture_etc") ||
245 isSupported(
"GL_OES_compressed_ETC1_RGB8_texture")) {
246 deviceCapabilities.TextureCompressionETC =
true;
248 if (isSupported(
"WEBGL_compressed_texture_pvrtc") ||
249 isSupported(
"GL_IMG_texture_compression_pvrtc")) {
250 deviceCapabilities.TextureCompressionPVRTC =
true;
252 if (isSupported(
"WEBGL_compressed_texture_astc") ||
253 isSupported(
"GL_OES_texture_compression_astc") || (
254 isSupported(
"GL_KHR_texture_compression_astc_hdr") &&
255 isSupported(
"GL_KHR_texture_compression_astc_ldr"))) {
256 deviceCapabilities.TextureCompressionASTC =
true;
259 OpenGLES30::enableExtensionFormats(
this);
262bool Cogs::CapabilitiesGLES30::isSupported(
const StringView & extension)
const
264 return extensions.find(std::string(extension)) != extensions.end();
Log implementation class.
constexpr Log getLogger(const char(&name)[LEN]) noexcept
uint32_t MaxTexture3DSize
Using D3D11_REQ_TEXTURE3D_U_V_OR_W_DIMENSION as default.
uint32_t MaxTexture2DSize
Using D3D11_REQ_TEXTURE2D_U_OR_V_DIMENSION as default.
uint32_t MaxTextureCubeSize
Using D3D11_REQ_TEXTURECUBE_DIMENSION as default.
uint32_t MaxTextureArrayLayers
Using D3D11_REQ_TEXTURE2D_ARRAY_AXIS_DIMENSION as default.
unsigned ConstantBufferOffsetAlignment
Minimum offset alignment when binding constant buffers.
bool ConstantBufferRange
supports binding a range of a constant buffer.
@ QuadroFX
nVidia Quadro FX professional graphics adapters.
@ FireGL
AMD FireGL series.
@ geForce
nVidia geForce consumer adapters.
@ Quadro
nVidia Quadro professional graphics adapters.
@ nVidia
nVidia Corporation.
@ Unknown
Unknown device vendor.