blob: 0efeac67cd9719811bab9ae5957e4affd275e683 [file] [log] [blame]
bsalomon@google.com27847de2011-02-22 20:59:41 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2011 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
bsalomon@google.com27847de2011-02-22 20:59:41 +00006 */
7
epoger@google.comec3ed6a2011-07-28 14:26:00 +00008
Mike Kleinc0bd9f92019-04-23 12:05:21 -05009#include "include/core/SkMatrix.h"
10#include "include/private/GrTypesPriv.h"
Robert Phillips8043f322019-05-31 08:11:36 -040011#include "src/gpu/GrDataUtils.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "src/gpu/gl/GrGLUtil.h"
bungeman@google.comfab44db2013-10-11 18:50:45 +000013#include <stdio.h>
bsalomon@google.comf987d1b2011-04-04 17:13:52 +000014
bsalomon@google.com0b77d682011-08-19 13:28:54 +000015void GrGLClearErr(const GrGLInterface* gl) {
commit-bot@chromium.orgc72425a2014-01-21 16:09:18 +000016 while (GR_GL_NO_ERROR != gl->fFunctions.fGetError()) {}
bsalomon@google.comf987d1b2011-04-04 17:13:52 +000017}
bsalomon@google.com27847de2011-02-22 20:59:41 +000018
bsalomon@google.com845eafd2012-06-18 12:27:29 +000019namespace {
20const char *get_error_string(uint32_t err) {
21 switch (err) {
22 case GR_GL_NO_ERROR:
23 return "";
24 case GR_GL_INVALID_ENUM:
25 return "Invalid Enum";
26 case GR_GL_INVALID_VALUE:
27 return "Invalid Value";
28 case GR_GL_INVALID_OPERATION:
29 return "Invalid Operation";
30 case GR_GL_OUT_OF_MEMORY:
31 return "Out of Memory";
32 case GR_GL_CONTEXT_LOST:
33 return "Context Lost";
34 }
35 return "Unknown";
36}
37}
38
bsalomon@google.com0b77d682011-08-19 13:28:54 +000039void GrGLCheckErr(const GrGLInterface* gl,
40 const char* location,
41 const char* call) {
42 uint32_t err = GR_GL_GET_ERROR(gl);
twiz@google.com0f31ca72011-03-18 17:38:11 +000043 if (GR_GL_NO_ERROR != err) {
tfarina38406c82014-10-31 07:11:12 -070044 SkDebugf("---- glGetError 0x%x(%s)", err, get_error_string(err));
bsalomon49f085d2014-09-05 13:34:00 -070045 if (location) {
tfarina38406c82014-10-31 07:11:12 -070046 SkDebugf(" at\n\t%s", location);
bsalomon@google.com27847de2011-02-22 20:59:41 +000047 }
bsalomon49f085d2014-09-05 13:34:00 -070048 if (call) {
tfarina38406c82014-10-31 07:11:12 -070049 SkDebugf("\n\t\t%s", call);
bsalomon@google.com27847de2011-02-22 20:59:41 +000050 }
tfarina38406c82014-10-31 07:11:12 -070051 SkDebugf("\n");
bsalomon@google.com27847de2011-02-22 20:59:41 +000052 }
53}
54
55///////////////////////////////////////////////////////////////////////////////
56
bsalomon@google.comd5d10492011-04-28 21:16:31 +000057#if GR_GL_LOG_CALLS
58 bool gLogCallsGL = !!(GR_GL_LOG_CALLS_START);
59#endif
bsalomon@google.com27847de2011-02-22 20:59:41 +000060
bsalomon@google.comd5d10492011-04-28 21:16:31 +000061#if GR_GL_CHECK_ERROR
62 bool gCheckErrorGL = !!(GR_GL_CHECK_ERROR_START);
63#endif
64
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +000065///////////////////////////////////////////////////////////////////////////////
66
commit-bot@chromium.org9e90aed2014-01-16 16:35:09 +000067GrGLStandard GrGLGetStandardInUseFromString(const char* versionString) {
halcanary96fcdcc2015-08-27 07:41:13 -070068 if (nullptr == versionString) {
69 SkDebugf("nullptr GL version string.");
commit-bot@chromium.org9e90aed2014-01-16 16:35:09 +000070 return kNone_GrGLStandard;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +000071 }
72
73 int major, minor;
74
75 // check for desktop
76 int n = sscanf(versionString, "%d.%d", &major, &minor);
77 if (2 == n) {
commit-bot@chromium.org9e90aed2014-01-16 16:35:09 +000078 return kGL_GrGLStandard;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +000079 }
80
Kevin Lubick39026282019-03-28 12:46:40 -040081 // WebGL might look like "OpenGL ES 2.0 (WebGL 1.0 (OpenGL ES 2.0 Chromium))"
82 int esMajor, esMinor;
83 n = sscanf(versionString, "OpenGL ES %d.%d (WebGL %d.%d", &esMajor, &esMinor, &major, &minor);
84 if (4 == n) {
85 return kWebGL_GrGLStandard;
86 }
87
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +000088 // check for ES 1
89 char profile[2];
bsalomon@google.com960d1142013-05-29 13:11:54 +000090 n = sscanf(versionString, "OpenGL ES-%c%c %d.%d", profile, profile+1, &major, &minor);
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +000091 if (4 == n) {
92 // we no longer support ES1.
commit-bot@chromium.org9e90aed2014-01-16 16:35:09 +000093 return kNone_GrGLStandard;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +000094 }
95
96 // check for ES2
97 n = sscanf(versionString, "OpenGL ES %d.%d", &major, &minor);
98 if (2 == n) {
commit-bot@chromium.org9e90aed2014-01-16 16:35:09 +000099 return kGLES_GrGLStandard;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000100 }
commit-bot@chromium.org9e90aed2014-01-16 16:35:09 +0000101 return kNone_GrGLStandard;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000102}
103
cdalton1acea862015-06-02 13:05:52 -0700104void GrGLGetDriverInfo(GrGLStandard standard,
105 GrGLVendor vendor,
106 const char* rendererString,
107 const char* versionString,
108 GrGLDriver* outDriver,
109 GrGLDriverVersion* outVersion) {
Brian Salomon9a544bc2018-04-04 16:12:31 -0400110 int major, minor, rev, driverMajor, driverMinor, driverPoint;
derekfa6ca5a02014-11-03 13:36:40 -0800111
cdalton1acea862015-06-02 13:05:52 -0700112 *outDriver = kUnknown_GrGLDriver;
113 *outVersion = GR_GL_DRIVER_UNKNOWN_VER;
bsalomon6383ac02016-03-23 14:55:21 -0700114 // These null checks are for test GL contexts that return nullptr in their
115 // glGetString implementation.
116 if (!rendererString) {
117 rendererString = "";
118 }
119 if (!versionString) {
120 versionString = "";
121 }
cdalton1acea862015-06-02 13:05:52 -0700122
Brian Salomona392dba2017-03-16 15:38:57 -0400123 static const char kChromium[] = "Chromium";
124 char suffix[SK_ARRAY_COUNT(kChromium)];
125 if (0 == strcmp(rendererString, kChromium) ||
126 (3 == sscanf(versionString, "OpenGL ES %d.%d %8s", &major, &minor, suffix) &&
127 0 == strcmp(kChromium, suffix))) {
cdalton1acea862015-06-02 13:05:52 -0700128 *outDriver = kChromium_GrGLDriver;
129 return;
130 }
derekfa6ca5a02014-11-03 13:36:40 -0800131
Kevin Lubick8aa203c2019-03-19 13:23:10 -0400132 if (GR_IS_GR_GL(standard)) {
cdalton1acea862015-06-02 13:05:52 -0700133 if (kNVIDIA_GrGLVendor == vendor) {
134 *outDriver = kNVIDIA_GrGLDriver;
135 int n = sscanf(versionString, "%d.%d.%d NVIDIA %d.%d",
136 &major, &minor, &rev, &driverMajor, &driverMinor);
137 // Some older NVIDIA drivers don't report the driver version.
138 if (5 == n) {
Brian Salomon9a544bc2018-04-04 16:12:31 -0400139 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, 0);
cdalton1acea862015-06-02 13:05:52 -0700140 }
141 return;
142 }
cdalton1acea862015-06-02 13:05:52 -0700143 int n = sscanf(versionString, "%d.%d Mesa %d.%d",
144 &major, &minor, &driverMajor, &driverMinor);
Chris Dalton06cd6662017-10-07 14:37:57 -0600145 if (4 != n) {
146 n = sscanf(versionString, "%d.%d (Core Profile) Mesa %d.%d",
147 &major, &minor, &driverMajor, &driverMinor);
148 }
cdalton1acea862015-06-02 13:05:52 -0700149 if (4 == n) {
150 *outDriver = kMesa_GrGLDriver;
Brian Salomon9a544bc2018-04-04 16:12:31 -0400151 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, 0);
cdalton1acea862015-06-02 13:05:52 -0700152 return;
153 }
Kevin Lubick8aa203c2019-03-19 13:23:10 -0400154 } else if (GR_IS_GR_GL_ES(standard)) {
cdalton1acea862015-06-02 13:05:52 -0700155 if (kNVIDIA_GrGLVendor == vendor) {
156 *outDriver = kNVIDIA_GrGLDriver;
157 int n = sscanf(versionString, "OpenGL ES %d.%d NVIDIA %d.%d",
158 &major, &minor, &driverMajor, &driverMinor);
159 // Some older NVIDIA drivers don't report the driver version.
160 if (4 == n) {
Brian Salomon9a544bc2018-04-04 16:12:31 -0400161 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, 0);
cdalton1acea862015-06-02 13:05:52 -0700162 }
163 return;
164 }
commit-bot@chromium.org459104c2013-06-14 14:42:56 +0000165
cdalton1acea862015-06-02 13:05:52 -0700166 int n = sscanf(versionString, "OpenGL ES %d.%d Mesa %d.%d",
167 &major, &minor, &driverMajor, &driverMinor);
168 if (4 == n) {
169 *outDriver = kMesa_GrGLDriver;
Brian Salomon9a544bc2018-04-04 16:12:31 -0400170 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, 0);
cdalton1acea862015-06-02 13:05:52 -0700171 return;
172 }
bsalomon88c7b982015-07-31 11:20:16 -0700173 if (0 == strncmp("ANGLE", rendererString, 5)) {
174 *outDriver = kANGLE_GrGLDriver;
175 n = sscanf(versionString, "OpenGL ES %d.%d (ANGLE %d.%d", &major, &minor, &driverMajor,
176 &driverMinor);
177 if (4 == n) {
Brian Salomon9a544bc2018-04-04 16:12:31 -0400178 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, 0);
bsalomon88c7b982015-07-31 11:20:16 -0700179 }
180 return;
181 }
cdalton1acea862015-06-02 13:05:52 -0700182 }
cdalton1dd05422015-06-12 09:01:18 -0700183
Kevin Lubickda2b9f32018-06-05 16:05:58 -0400184 if (kGoogle_GrGLVendor == vendor) {
185 // Swiftshader is the only Google vendor at the moment
186 *outDriver = kSwiftShader_GrGLDriver;
187
188 // Swiftshader has a strange version string: w.x.y.z Going to arbitrarily ignore
189 // y and assume w,x and z are major, minor, point.
190 // As of writing, version is 4.0.0.6
191 int n = sscanf(versionString, "OpenGL ES %d.%d SwiftShader %d.%d.0.%d", &major, &minor,
192 &driverMajor, &driverMinor, &driverPoint);
193 if (5 == n) {
194 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, driverPoint);
195 }
196 return;
197 }
198
cdalton1dd05422015-06-12 09:01:18 -0700199 if (kIntel_GrGLVendor == vendor) {
200 // We presume we're on the Intel driver since it hasn't identified itself as Mesa.
201 *outDriver = kIntel_GrGLDriver;
Brian Salomon9a544bc2018-04-04 16:12:31 -0400202
203 //This is how the macOS version strings are structured. This might be different on different
204 // OSes.
205 int n = sscanf(versionString, "%d.%d INTEL-%d.%d.%d", &major, &minor, &driverMajor,
206 &driverMinor, &driverPoint);
207 if (5 == n) {
208 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, driverPoint);
209 }
cdalton1dd05422015-06-12 09:01:18 -0700210 }
Brian Salomonb52fa022017-06-07 09:42:52 -0400211
212 if (kQualcomm_GrGLVendor == vendor) {
213 *outDriver = kQualcomm_GrGLDriver;
214 int n = sscanf(versionString, "OpenGL ES %d.%d V@%d.%d", &major, &minor, &driverMajor,
215 &driverMinor);
216 if (4 == n) {
Brian Salomon9a544bc2018-04-04 16:12:31 -0400217 *outVersion = GR_GL_DRIVER_VER(driverMajor, driverMinor, 0);
Brian Salomonb52fa022017-06-07 09:42:52 -0400218 }
219 return;
220 }
commit-bot@chromium.orgc9424b82013-10-30 20:03:16 +0000221}
222
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000223GrGLVersion GrGLGetVersionFromString(const char* versionString) {
halcanary96fcdcc2015-08-27 07:41:13 -0700224 if (nullptr == versionString) {
225 SkDebugf("nullptr GL version string.");
commit-bot@chromium.orgf4e67e32014-04-30 01:26:04 +0000226 return GR_GL_INVALID_VER;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000227 }
228
229 int major, minor;
230
bsalomon@google.com960d1142013-05-29 13:11:54 +0000231 // check for mesa
232 int mesaMajor, mesaMinor;
233 int n = sscanf(versionString, "%d.%d Mesa %d.%d", &major, &minor, &mesaMajor, &mesaMinor);
234 if (4 == n) {
joshualitt46821702014-07-30 11:49:12 -0700235 return GR_GL_VER(major, minor);
bsalomon@google.com960d1142013-05-29 13:11:54 +0000236 }
237
238 n = sscanf(versionString, "%d.%d", &major, &minor);
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000239 if (2 == n) {
240 return GR_GL_VER(major, minor);
241 }
242
Kevin Lubick39026282019-03-28 12:46:40 -0400243 // WebGL might look like "OpenGL ES 2.0 (WebGL 1.0 (OpenGL ES 2.0 Chromium))"
244 int esMajor, esMinor;
245 n = sscanf(versionString, "OpenGL ES %d.%d (WebGL %d.%d", &esMajor, &esMinor, &major, &minor);
246 if (4 == n) {
247 return GR_GL_VER(major, minor);
248 }
249
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000250 char profile[2];
251 n = sscanf(versionString, "OpenGL ES-%c%c %d.%d", profile, profile+1,
252 &major, &minor);
253 if (4 == n) {
254 return GR_GL_VER(major, minor);
255 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000256
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000257 n = sscanf(versionString, "OpenGL ES %d.%d", &major, &minor);
258 if (2 == n) {
259 return GR_GL_VER(major, minor);
260 }
261
commit-bot@chromium.orgf4e67e32014-04-30 01:26:04 +0000262 return GR_GL_INVALID_VER;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000263}
264
265GrGLSLVersion GrGLGetGLSLVersionFromString(const char* versionString) {
halcanary96fcdcc2015-08-27 07:41:13 -0700266 if (nullptr == versionString) {
267 SkDebugf("nullptr GLSL version string.");
commit-bot@chromium.orgf4e67e32014-04-30 01:26:04 +0000268 return GR_GLSL_INVALID_VER;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000269 }
270
271 int major, minor;
272
273 int n = sscanf(versionString, "%d.%d", &major, &minor);
274 if (2 == n) {
275 return GR_GLSL_VER(major, minor);
276 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000277
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000278 n = sscanf(versionString, "OpenGL ES GLSL ES %d.%d", &major, &minor);
279 if (2 == n) {
280 return GR_GLSL_VER(major, minor);
281 }
282
283#ifdef SK_BUILD_FOR_ANDROID
284 // android hack until the gpu vender updates their drivers
285 n = sscanf(versionString, "OpenGL ES GLSL %d.%d", &major, &minor);
286 if (2 == n) {
287 return GR_GLSL_VER(major, minor);
288 }
289#endif
290
commit-bot@chromium.orgf4e67e32014-04-30 01:26:04 +0000291 return GR_GLSL_INVALID_VER;
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000292}
293
bsalomon@google.com0b1e4812012-10-23 13:52:43 +0000294GrGLVendor GrGLGetVendorFromString(const char* vendorString) {
bsalomon49f085d2014-09-05 13:34:00 -0700295 if (vendorString) {
bsalomon@google.com96966a52013-02-21 16:34:21 +0000296 if (0 == strcmp(vendorString, "ARM")) {
297 return kARM_GrGLVendor;
298 }
Kevin Lubickda2b9f32018-06-05 16:05:58 -0400299 if (0 == strcmp(vendorString, "Google Inc.")) {
300 return kGoogle_GrGLVendor;
301 }
bsalomon@google.com3012ded2013-02-22 16:44:04 +0000302 if (0 == strcmp(vendorString, "Imagination Technologies")) {
303 return kImagination_GrGLVendor;
304 }
commit-bot@chromium.org54318d32014-02-14 17:27:04 +0000305 if (0 == strncmp(vendorString, "Intel ", 6) || 0 == strcmp(vendorString, "Intel")) {
bsalomon@google.com3012ded2013-02-22 16:44:04 +0000306 return kIntel_GrGLVendor;
307 }
commit-bot@chromium.org7a434a22013-08-21 14:01:56 +0000308 if (0 == strcmp(vendorString, "Qualcomm")) {
309 return kQualcomm_GrGLVendor;
310 }
bsalomone904c092014-07-17 10:50:59 -0700311 if (0 == strcmp(vendorString, "NVIDIA Corporation")) {
bsalomon63b21962014-11-05 07:05:34 -0800312 return kNVIDIA_GrGLVendor;
bsalomone904c092014-07-17 10:50:59 -0700313 }
brianosman131ff132016-06-07 14:22:44 -0700314 if (0 == strcmp(vendorString, "ATI Technologies Inc.")) {
315 return kATI_GrGLVendor;
316 }
bsalomon@google.com0b1e4812012-10-23 13:52:43 +0000317 }
bsalomon@google.com0b1e4812012-10-23 13:52:43 +0000318 return kOther_GrGLVendor;
319}
320
Brian Salomon266ef6d2017-09-22 11:27:42 -0400321static bool is_renderer_angle(const char* rendererString) {
322 static constexpr char kHeader[] = "ANGLE ";
323 static constexpr size_t kHeaderLength = SK_ARRAY_COUNT(kHeader) - 1;
Brian Salomon2ab5e782018-12-18 09:00:36 -0500324 return rendererString && 0 == strncmp(rendererString, kHeader, kHeaderLength);
Brian Salomon266ef6d2017-09-22 11:27:42 -0400325}
326
Brian Salomon4470e342018-04-04 14:27:48 -0400327GrGLRenderer GrGLGetRendererFromStrings(const char* rendererString,
328 const GrGLExtensions& extensions) {
bsalomon49f085d2014-09-05 13:34:00 -0700329 if (rendererString) {
Chris Dalton0090ef62018-03-28 17:35:00 -0600330 static const char kTegraStr[] = "NVIDIA Tegra";
331 if (0 == strncmp(rendererString, kTegraStr, SK_ARRAY_COUNT(kTegraStr) - 1)) {
332 // Tegra strings are not very descriptive. We distinguish between the modern and legacy
333 // architectures by the presence of NV_path_rendering.
Brian Salomon4470e342018-04-04 14:27:48 -0400334 return extensions.has("GL_NV_path_rendering") ? kTegra_GrGLRenderer
335 : kTegra_PreK1_GrGLRenderer;
commit-bot@chromium.org0694ea72013-09-18 13:00:28 +0000336 }
bsalomon63b21962014-11-05 07:05:34 -0800337 int lastDigit;
338 int n = sscanf(rendererString, "PowerVR SGX 54%d", &lastDigit);
339 if (1 == n && lastDigit >= 0 && lastDigit <= 9) {
340 return kPowerVR54x_GrGLRenderer;
341 }
jvanverthf10ecb72015-12-10 09:06:24 -0800342 // certain iOS devices also use PowerVR54x GPUs
343 static const char kAppleA4Str[] = "Apple A4";
344 static const char kAppleA5Str[] = "Apple A5";
345 static const char kAppleA6Str[] = "Apple A6";
346 if (0 == strncmp(rendererString, kAppleA4Str,
347 SK_ARRAY_COUNT(kAppleA4Str)-1) ||
348 0 == strncmp(rendererString, kAppleA5Str,
349 SK_ARRAY_COUNT(kAppleA5Str)-1) ||
350 0 == strncmp(rendererString, kAppleA6Str,
351 SK_ARRAY_COUNT(kAppleA6Str)-1)) {
352 return kPowerVR54x_GrGLRenderer;
353 }
bsalomon63b21962014-11-05 07:05:34 -0800354 static const char kPowerVRRogueStr[] = "PowerVR Rogue";
jvanverthf10ecb72015-12-10 09:06:24 -0800355 static const char kAppleA7Str[] = "Apple A7";
356 static const char kAppleA8Str[] = "Apple A8";
bsalomon63b21962014-11-05 07:05:34 -0800357 if (0 == strncmp(rendererString, kPowerVRRogueStr,
jvanverthf10ecb72015-12-10 09:06:24 -0800358 SK_ARRAY_COUNT(kPowerVRRogueStr)-1) ||
359 0 == strncmp(rendererString, kAppleA7Str,
360 SK_ARRAY_COUNT(kAppleA7Str)-1) ||
361 0 == strncmp(rendererString, kAppleA8Str,
362 SK_ARRAY_COUNT(kAppleA8Str)-1)) {
bsalomon63b21962014-11-05 07:05:34 -0800363 return kPowerVRRogue_GrGLRenderer;
364 }
bsalomona8fcea02015-02-13 09:00:39 -0800365 int adrenoNumber;
366 n = sscanf(rendererString, "Adreno (TM) %d", &adrenoNumber);
367 if (1 == n) {
368 if (adrenoNumber >= 300) {
369 if (adrenoNumber < 400) {
370 return kAdreno3xx_GrGLRenderer;
371 }
372 if (adrenoNumber < 500) {
Chris Dalton1214be92018-06-28 19:06:27 -0600373 return adrenoNumber >= 430
374 ? kAdreno430_GrGLRenderer : kAdreno4xx_other_GrGLRenderer;
bsalomona8fcea02015-02-13 09:00:39 -0800375 }
egdaniel7517e452016-09-20 13:00:26 -0700376 if (adrenoNumber < 600) {
377 return kAdreno5xx_GrGLRenderer;
378 }
bsalomona8fcea02015-02-13 09:00:39 -0800379 }
380 }
Kevin Lubickda2b9f32018-06-05 16:05:58 -0400381 if (0 == strcmp("Google SwiftShader", rendererString)) {
382 return kGoogleSwiftShader_GrGLRenderer;
383 }
Robert Phillipsa2fd62a2017-10-05 12:46:21 -0400384
Chris Daltonb70beea2018-07-23 12:17:10 -0600385 if (const char* intelString = strstr(rendererString, "Intel")) {
Brian Osmand6b53d52019-06-25 15:37:14 -0400386 // These generic strings seem to always come from Haswell: Iris 5100 or Iris Pro 5200
387 if (0 == strcmp("Intel Iris OpenGL Engine", intelString) ||
388 0 == strcmp("Intel Iris Pro OpenGL Engine", intelString)) {
389 return kIntelHaswell_GrGLRenderer;
Chris Dalton4a6a7322017-10-06 14:28:11 -0600390 }
Chris Daltonb70beea2018-07-23 12:17:10 -0600391 if (strstr(intelString, "Sandybridge")) {
Chris Dalton1c1391c2018-07-20 12:21:14 -0600392 return kIntelSandyBridge_GrGLRenderer;
393 }
Chris Daltonb70beea2018-07-23 12:17:10 -0600394 if (strstr(intelString, "Bay Trail")) {
Brian Osmand6b53d52019-06-25 15:37:14 -0400395 return kIntelValleyView_GrGLRenderer;
Chris Daltonb70beea2018-07-23 12:17:10 -0600396 }
Brian Osmand6b53d52019-06-25 15:37:14 -0400397 // There are many possible intervening strings here:
398 // 'Intel(R)' is a common prefix
399 // 'Iris' may appear, followed by '(R)' or '(TM)'
400 // 'Iris' can then be followed by 'Graphics', 'Pro Graphics', or 'Plus Graphics'
401 // If 'Iris' isn't there, we might have 'HD Graphics' or 'UHD Graphics'
402 //
403 // In all cases, though, we end with 'Graphics ', an optional 'P', and a number,
404 // so just skip to that and handle two cases:
405 if (const char* intelGfxString = strstr(intelString, "Graphics")) {
406 int intelNumber;
407 if (sscanf(intelGfxString, "Graphics %d", &intelNumber) ||
408 sscanf(intelGfxString, "Graphics P%d", &intelNumber)) {
Chris Daltonb70beea2018-07-23 12:17:10 -0600409
Brian Osmand6b53d52019-06-25 15:37:14 -0400410 if (intelNumber == 2000 || intelNumber == 3000) {
411 return kIntelSandyBridge_GrGLRenderer;
412 }
413 if (intelNumber == 2500 || intelNumber == 4000) {
414 return kIntelIvyBridge_GrGLRenderer;
415 }
416 if (intelNumber >= 4200 && intelNumber <= 5200) {
417 return kIntelHaswell_GrGLRenderer;
418 }
419 if (intelNumber >= 400 && intelNumber <= 405) {
420 return kIntelCherryView_GrGLRenderer;
421 }
422 if (intelNumber >= 5300 && intelNumber <= 6300) {
423 return kIntelBroadwell_GrGLRenderer;
424 }
425 if (intelNumber >= 500 && intelNumber <= 505) {
426 return kIntelApolloLake_GrGLRenderer;
427 }
428 if (intelNumber >= 510 && intelNumber <= 580) {
429 return kIntelSkyLake_GrGLRenderer;
430 }
431 if (intelNumber >= 600 && intelNumber <= 605) {
432 return kIntelGeminiLake_GrGLRenderer;
433 }
434 // 610 and 630 are reused from KabyLake to CoffeeLake. The CoffeeLake variants
435 // are "UHD Graphics", while the KabyLake ones are "HD Graphics"
436 if (intelNumber == 610 || intelNumber == 630) {
437 return strstr(intelString, "UHD") ? kIntelCoffeeLake_GrGLRenderer
438 : kIntelKabyLake_GrGLRenderer;
439 }
440 if (intelNumber >= 610 && intelNumber <= 650) {
441 return kIntelKabyLake_GrGLRenderer;
442 }
443 if (intelNumber == 655) {
444 return kIntelCoffeeLake_GrGLRenderer;
445 }
446 if (intelNumber >= 910 && intelNumber <= 950) {
447 return kIntelIceLake_GrGLRenderer;
448 }
Chris Daltonb70beea2018-07-23 12:17:10 -0600449 }
Chris Daltonda40cd22018-04-16 13:19:58 -0600450 }
Brian Salomon028a9a52017-05-11 11:39:08 -0400451 }
Robert Phillipsdbfecd02017-10-18 15:44:08 -0400452
453 // The AMD string can have a somewhat arbitrary preamble (see skbug.com/7195)
454 if (const char* amdString = strstr(rendererString, "Radeon")) {
455 char amdGeneration, amdTier, amdRevision;
456 n = sscanf(amdString, "Radeon (TM) R9 M%c%c%c",
457 &amdGeneration, &amdTier, &amdRevision);
458 if (3 == n) {
459 if ('4' == amdGeneration) {
460 return kAMDRadeonR9M4xx_GrGLRenderer;
461 }
462 }
463
464 char amd0, amd1, amd2;
465 n = sscanf(amdString, "Radeon HD 7%c%c%c Series", &amd0, &amd1, &amd2);
466 if (3 == n) {
467 return kAMDRadeonHD7xxx_GrGLRenderer;
468 }
469 }
470
Brian Salomon0ee6f952017-01-19 15:52:24 -0500471 if (0 == strcmp("Mesa Offscreen", rendererString)) {
bsalomon40170072016-05-05 14:40:03 -0700472 return kOSMesa_GrGLRenderer;
473 }
Greg Daniel25019172017-10-26 13:32:33 -0400474 if (strstr(rendererString, "llvmpipe")) {
475 return kGalliumLLVM_GrGLRenderer;
476 }
Brian Osmanac1e4962017-05-25 11:34:38 -0400477 static const char kMaliTStr[] = "Mali-T";
478 if (0 == strncmp(rendererString, kMaliTStr, SK_ARRAY_COUNT(kMaliTStr) - 1)) {
479 return kMaliT_GrGLRenderer;
480 }
Brian Salomon5fba7ad2018-03-22 10:01:16 -0400481 int mali400Num;
482 if (1 == sscanf(rendererString, "Mali-%d", &mali400Num) && mali400Num >= 400 &&
483 mali400Num < 500) {
484 return kMali4xx_GrGLRenderer;
485 }
Brian Salomon266ef6d2017-09-22 11:27:42 -0400486 if (is_renderer_angle(rendererString)) {
Robert Phillipsbc712742017-06-02 11:20:37 -0400487 return kANGLE_GrGLRenderer;
488 }
commit-bot@chromium.org0694ea72013-09-18 13:00:28 +0000489 }
490 return kOther_GrGLRenderer;
491}
492
Brian Salomon266ef6d2017-09-22 11:27:42 -0400493void GrGLGetANGLEInfoFromString(const char* rendererString, GrGLANGLEBackend* backend,
494 GrGLANGLEVendor* vendor, GrGLANGLERenderer* renderer) {
495 *backend = GrGLANGLEBackend::kUnknown;
496 *vendor = GrGLANGLEVendor::kUnknown;
497 *renderer = GrGLANGLERenderer::kUnknown;
498 if (!is_renderer_angle(rendererString)) {
499 return;
500 }
501 if (strstr(rendererString, "Intel")) {
502 *vendor = GrGLANGLEVendor::kIntel;
Chris Dalton0a94e4c2018-01-18 15:06:50 -0700503
504 const char* modelStr;
505 int modelNumber;
506 if ((modelStr = strstr(rendererString, "HD Graphics")) &&
507 (1 == sscanf(modelStr, "HD Graphics %i", &modelNumber) ||
508 1 == sscanf(modelStr, "HD Graphics P%i", &modelNumber))) {
509 switch (modelNumber) {
Chris Dalton1c1391c2018-07-20 12:21:14 -0600510 case 2000:
511 case 3000:
512 *renderer = GrGLANGLERenderer::kSandyBridge;
513 break;
Chris Dalton0a94e4c2018-01-18 15:06:50 -0700514 case 4000:
515 case 2500:
516 *renderer = GrGLANGLERenderer::kIvyBridge;
517 break;
518 case 510:
519 case 515:
520 case 520:
521 case 530:
522 *renderer = GrGLANGLERenderer::kSkylake;
523 break;
524 }
525 } else if ((modelStr = strstr(rendererString, "Iris")) &&
526 (1 == sscanf(modelStr, "Iris(TM) Graphics %i", &modelNumber) ||
527 1 == sscanf(modelStr, "Iris(TM) Pro Graphics %i", &modelNumber) ||
528 1 == sscanf(modelStr, "Iris(TM) Pro Graphics P%i", &modelNumber))) {
529 switch (modelNumber) {
530 case 540:
531 case 550:
532 case 555:
533 case 580:
534 *renderer = GrGLANGLERenderer::kSkylake;
535 break;
536 }
537 }
Brian Salomon266ef6d2017-09-22 11:27:42 -0400538 }
539 if (strstr(rendererString, "Direct3D11")) {
540 *backend = GrGLANGLEBackend::kD3D11;
541 } else if (strstr(rendererString, "Direct3D9")) {
542 *backend = GrGLANGLEBackend::kD3D9;
543 } else if (strstr(rendererString, "OpenGL")) {
544 *backend = GrGLANGLEBackend::kOpenGL;
545 }
546}
547
bsalomon@google.com9c1f1ac2012-05-07 17:09:37 +0000548GrGLVersion GrGLGetVersion(const GrGLInterface* gl) {
549 const GrGLubyte* v;
550 GR_GL_CALL_RET(gl, v, GetString(GR_GL_VERSION));
551 return GrGLGetVersionFromString((const char*) v);
552}
553
554GrGLSLVersion GrGLGetGLSLVersion(const GrGLInterface* gl) {
555 const GrGLubyte* v;
556 GR_GL_CALL_RET(gl, v, GetString(GR_GL_SHADING_LANGUAGE_VERSION));
557 return GrGLGetGLSLVersionFromString((const char*) v);
558}
bsalomon@google.com0b1e4812012-10-23 13:52:43 +0000559
560GrGLVendor GrGLGetVendor(const GrGLInterface* gl) {
561 const GrGLubyte* v;
562 GR_GL_CALL_RET(gl, v, GetString(GR_GL_VENDOR));
563 return GrGLGetVendorFromString((const char*) v);
564}
commit-bot@chromium.org215a6822013-09-05 18:28:42 +0000565
commit-bot@chromium.org0694ea72013-09-18 13:00:28 +0000566GrGLRenderer GrGLGetRenderer(const GrGLInterface* gl) {
Chris Dalton0090ef62018-03-28 17:35:00 -0600567 const GrGLubyte* rendererString;
568 GR_GL_CALL_RET(gl, rendererString, GetString(GR_GL_RENDERER));
569
Brian Salomon4470e342018-04-04 14:27:48 -0400570 return GrGLGetRendererFromStrings((const char*)rendererString, gl->fExtensions);
commit-bot@chromium.org0694ea72013-09-18 13:00:28 +0000571}
572
cdalton93a379b2016-05-11 13:58:08 -0700573GrGLenum GrToGLStencilFunc(GrStencilTest test) {
574 static const GrGLenum gTable[kGrStencilTestCount] = {
575 GR_GL_ALWAYS, // kAlways
576 GR_GL_NEVER, // kNever
577 GR_GL_GREATER, // kGreater
578 GR_GL_GEQUAL, // kGEqual
579 GR_GL_LESS, // kLess
580 GR_GL_LEQUAL, // kLEqual
581 GR_GL_EQUAL, // kEqual
582 GR_GL_NOTEQUAL, // kNotEqual
kkinnunenccdaa042014-08-20 01:36:23 -0700583 };
cdalton93a379b2016-05-11 13:58:08 -0700584 GR_STATIC_ASSERT(0 == (int)GrStencilTest::kAlways);
585 GR_STATIC_ASSERT(1 == (int)GrStencilTest::kNever);
586 GR_STATIC_ASSERT(2 == (int)GrStencilTest::kGreater);
587 GR_STATIC_ASSERT(3 == (int)GrStencilTest::kGEqual);
588 GR_STATIC_ASSERT(4 == (int)GrStencilTest::kLess);
589 GR_STATIC_ASSERT(5 == (int)GrStencilTest::kLEqual);
590 GR_STATIC_ASSERT(6 == (int)GrStencilTest::kEqual);
591 GR_STATIC_ASSERT(7 == (int)GrStencilTest::kNotEqual);
592 SkASSERT(test < (GrStencilTest)kGrStencilTestCount);
kkinnunenccdaa042014-08-20 01:36:23 -0700593
cdalton93a379b2016-05-11 13:58:08 -0700594 return gTable[(int)test];
kkinnunenccdaa042014-08-20 01:36:23 -0700595}
Robert Phillips8043f322019-05-31 08:11:36 -0400596
597bool GrGLFormatIsCompressed(GrGLenum glFormat) {
598 switch (glFormat) {
599 case GR_GL_COMPRESSED_RGB8_ETC2: // fall through
600 case GR_GL_COMPRESSED_ETC1_RGB8:
601 return true;
602 default:
603 return false;
604 }
Robert Phillips8043f322019-05-31 08:11:36 -0400605}
606
Brian Salomonbb8dde82019-06-27 10:52:13 -0400607bool GrGLFormatToCompressionType(GrGLenum glFormat, SkImage::CompressionType* compressionType) {
Robert Phillips28a5a432019-06-07 12:46:21 -0400608 switch (glFormat) {
609 case GR_GL_COMPRESSED_RGB8_ETC2: // fall through
610 case GR_GL_COMPRESSED_ETC1_RGB8:
Brian Salomonbb8dde82019-06-27 10:52:13 -0400611 *compressionType = SkImage::kETC1_CompressionType;
612 return true;
Robert Phillips28a5a432019-06-07 12:46:21 -0400613 default:
Brian Salomonbb8dde82019-06-27 10:52:13 -0400614 return false;
Robert Phillips28a5a432019-06-07 12:46:21 -0400615 }
Robert Phillips8043f322019-05-31 08:11:36 -0400616}
617
Robert Phillipsfe18de52019-06-06 17:21:50 -0400618size_t GrGLBytesPerFormat(GrGLenum glFormat) {
619 switch (glFormat) {
620 case GR_GL_LUMINANCE8:
621 case GR_GL_ALPHA8:
622 case GR_GL_R8:
623 return 1;
624
625 case GR_GL_RGB565:
626 case GR_GL_RGBA4:
627 case GR_GL_RG8:
628 case GR_GL_R16F:
Robert Phillips66a46032019-06-18 08:00:42 -0400629 case GR_GL_R16:
Robert Phillipsfe18de52019-06-06 17:21:50 -0400630 return 2;
631
632 case GR_GL_RGB8:
633 return 3;
634
635 case GR_GL_RGBA8:
636 case GR_GL_SRGB8_ALPHA8:
637 case GR_GL_BGRA8:
638 case GR_GL_RGB10_A2:
Robert Phillips66a46032019-06-18 08:00:42 -0400639 case GR_GL_RG16:
Robert Phillipsfe18de52019-06-06 17:21:50 -0400640 return 4;
641
642 case GR_GL_RGBA16F:
643 case GR_GL_RG32F:
644 return 8;
645
646 case GR_GL_RGBA32F:
647 return 16;
648
649 case GR_GL_COMPRESSED_RGB8_ETC2: // fall through
650 case GR_GL_COMPRESSED_ETC1_RGB8:
651 return 0;
652
Robert Phillips66a46032019-06-18 08:00:42 -0400653 // Experimental (for Y416 and mutant P016/P010)
654 case GR_GL_RG16F:
Robert Phillipsfe18de52019-06-06 17:21:50 -0400655 return 4;
Robert Phillips66a46032019-06-18 08:00:42 -0400656 case GR_GL_RGBA16:
657 return 8;
Robert Phillipsfe18de52019-06-06 17:21:50 -0400658 }
659
660 SK_ABORT("Invalid GL format");
661 return 0;
662}
Robert Phillipsf209e882019-06-25 15:59:50 -0400663
664bool GrGLFormatIsSupported(GrGLenum format) {
665 switch (format) {
666 case GR_GL_LUMINANCE8:
667 case GR_GL_ALPHA8:
668 case GR_GL_R8:
669 case GR_GL_RGB565:
670 case GR_GL_RGBA4:
671 case GR_GL_RG8:
672 case GR_GL_R16F:
673 case GR_GL_R16:
674 case GR_GL_RGB8:
675 case GR_GL_RGBA8:
676 case GR_GL_SRGB8_ALPHA8:
677 case GR_GL_BGRA8:
678 case GR_GL_RGB10_A2:
679 case GR_GL_RG16:
680 case GR_GL_RGBA16F:
681 case GR_GL_RG32F:
682 case GR_GL_RGBA32F:
683 case GR_GL_COMPRESSED_RGB8_ETC2:
684 case GR_GL_COMPRESSED_ETC1_RGB8:
685 // Experimental (for Y416 and mutant P016/P010)
686 case GR_GL_RG16F:
687 case GR_GL_RGBA16:
688 return true;
689 default:
690 return false;
691 }
692}