15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "gpu/command_buffer/service/feature_info.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <set>
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/command_line.h"
105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)#include "base/macros.h"
110529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch#include "base/metrics/histogram.h"
12c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#include "base/strings/string_number_conversions.h"
13c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#include "base/strings/string_split.h"
14868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/string_util.h"
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "gpu/command_buffer/service/gl_utils.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "gpu/command_buffer/service/gpu_switches.h"
17116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch#include "ui/gl/gl_fence.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ui/gl/gl_implementation.h"
197d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace gpu {
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace gles2 {
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)struct FormatInfo {
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  GLenum format;
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const GLenum* types;
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  size_t count;
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)class StringSet {
325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StringSet() {}
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StringSet(const char* s) {
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Init(s);
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  StringSet(const std::string& str) {
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Init(str);
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Init(const char* s) {
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    std::string str(s ? s : "");
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Init(str);
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Init(const std::string& str) {
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    std::vector<std::string> tokens;
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    Tokenize(str, " ", &tokens);
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    string_set_.insert(tokens.begin(), tokens.end());
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool Contains(const char* s) {
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return string_set_.find(s) != string_set_.end();
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool Contains(const std::string& s) {
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return string_set_.find(s) != string_set_.end();
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::set<std::string> string_set_;
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
66c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// Process a string of wordaround type IDs (seperated by ',') and set up
67c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// the corresponding Workaround flags.
68c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void StringToWorkarounds(
69c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    const std::string& types, FeatureInfo::Workarounds* workarounds) {
70c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  DCHECK(workarounds);
71c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  std::vector<std::string> pieces;
72c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  base::SplitString(types, ',', &pieces);
73c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  for (size_t i = 0; i < pieces.size(); ++i) {
74c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    int number = 0;
75c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    bool succeed = base::StringToInt(pieces[i], &number);
76c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    DCHECK(succeed);
77c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    switch (number) {
78c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#define GPU_OP(type, name)    \
79c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  case gpu::type:             \
80c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    workarounds->name = true; \
81c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    break;
82c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      GPU_DRIVER_BUG_WORKAROUNDS(GPU_OP)
83c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#undef GPU_OP
84c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      default:
85c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        NOTIMPLEMENTED();
86c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    }
87c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
88c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (workarounds->max_texture_size_limit_4096)
89c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    workarounds->max_texture_size = 4096;
90c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (workarounds->max_cube_map_texture_size_limit_4096)
91c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    workarounds->max_cube_map_texture_size = 4096;
92c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (workarounds->max_cube_map_texture_size_limit_1024)
93c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    workarounds->max_cube_map_texture_size = 1024;
94c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (workarounds->max_cube_map_texture_size_limit_512)
95c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    workarounds->max_cube_map_texture_size = 512;
960529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch
970529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  if (workarounds->max_fragment_uniform_vectors_32)
980529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    workarounds->max_fragment_uniform_vectors = 32;
990529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  if (workarounds->max_varying_vectors_16)
1000529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    workarounds->max_varying_vectors = 16;
1010529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  if (workarounds->max_vertex_uniform_vectors_256)
1020529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    workarounds->max_vertex_uniform_vectors = 256;
103c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)}
104c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // anonymous namespace.
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)FeatureInfo::FeatureFlags::FeatureFlags()
1085d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    : chromium_color_buffer_float_rgba(false),
1095d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      chromium_color_buffer_float_rgb(false),
1105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      chromium_framebuffer_multisample(false),
1110529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      chromium_sync_query(false),
112f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      use_core_framebuffer_multisample(false),
1137dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      multisampled_render_to_texture(false),
1147dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      use_img_for_multisampled_render_to_texture(false),
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      oes_standard_derivatives(false),
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      oes_egl_image_external(false),
11768043e1e95eeb07d5cae7aca370b26518b0867d6Torne (Richard Coles)      oes_depth24(false),
118a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)      oes_compressed_etc1_rgb8_texture(false),
119f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      packed_depth24_stencil8(false),
1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      npot_ok(false),
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      enable_texture_float_linear(false),
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      enable_texture_half_float_linear(false),
1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      angle_translated_shader_source(false),
1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      angle_pack_reverse_row_order(false),
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      arb_texture_rectangle(false),
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      angle_instanced_arrays(false),
1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      occlusion_query_boolean(false),
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      use_arb_occlusion_query2_for_occlusion_query_boolean(false),
1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      use_arb_occlusion_query_for_occlusion_query_boolean(false),
1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      native_vertex_array_object(false),
131a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)      ext_texture_format_bgra8888(false),
1322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      enable_shader_name_hashing(false),
1332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      enable_samplers(false),
134868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      ext_draw_buffers(false),
135a3f7b4e666c476898878fa745f637129375cd889Ben Murdoch      ext_frag_depth(false),
1360529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      ext_shader_texture_lod(false),
137424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)      use_async_readpixels(false),
1380f1bc08d4cfcc34181b0b5cbf065c40f687bf740Torne (Richard Coles)      map_buffer_range(false),
139f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      ext_discard_framebuffer(false),
140f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      angle_depth_texture(false),
141f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      is_angle(false),
142a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)      is_swiftshader(false),
143a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)      angle_texture_usage(false),
1441320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      ext_texture_storage(false),
1451320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      chromium_path_rendering(false) {
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
148c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)FeatureInfo::Workarounds::Workarounds() :
149c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#define GPU_OP(type, name) name(false),
150c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    GPU_DRIVER_BUG_WORKAROUNDS(GPU_OP)
151c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#undef GPU_OP
152c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    max_texture_size(0),
1530529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    max_cube_map_texture_size(0),
1540529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    max_fragment_uniform_vectors(0),
1550529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    max_varying_vectors(0),
1560529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    max_vertex_uniform_vectors(0) {
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)FeatureInfo::FeatureInfo() {
16058537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  InitializeBasicState(*CommandLine::ForCurrentProcess());
16158537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)}
16258537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)
16358537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)FeatureInfo::FeatureInfo(const CommandLine& command_line) {
16458537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  InitializeBasicState(command_line);
16558537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)}
16658537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)
16758537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)void FeatureInfo::InitializeBasicState(const CommandLine& command_line) {
16858537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  if (command_line.HasSwitch(switches::kGpuDriverBugWorkarounds)) {
16958537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    std::string types = command_line.GetSwitchValueASCII(
17058537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)        switches::kGpuDriverBugWorkarounds);
17158537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    StringToWorkarounds(types, &workarounds_);
17258537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  }
17358537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  feature_flags_.enable_shader_name_hashing =
17458537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      !command_line.HasSwitch(switches::kDisableShaderNameHashing);
17558537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)
176f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  feature_flags_.is_swiftshader =
177f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      (command_line.GetSwitchValueASCII(switches::kUseGL) == "swiftshader");
178f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)
1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static const GLenum kAlphaTypes[] = {
1805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_BYTE,
1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
1825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static const GLenum kRGBTypes[] = {
1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_BYTE,
1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_SHORT_5_6_5,
1855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
1865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static const GLenum kRGBATypes[] = {
1875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_BYTE,
1885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_SHORT_4_4_4_4,
1895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_SHORT_5_5_5_1,
1905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
1915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static const GLenum kLuminanceTypes[] = {
1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_BYTE,
1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
1945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static const GLenum kLuminanceAlphaTypes[] = {
1955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      GL_UNSIGNED_BYTE,
1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
1975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static const FormatInfo kFormatTypes[] = {
1985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    { GL_ALPHA, kAlphaTypes, arraysize(kAlphaTypes), },
1995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    { GL_RGB, kRGBTypes, arraysize(kRGBTypes), },
2005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    { GL_RGBA, kRGBATypes, arraysize(kRGBATypes), },
2015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    { GL_LUMINANCE, kLuminanceTypes, arraysize(kLuminanceTypes), },
2025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    { GL_LUMINANCE_ALPHA, kLuminanceAlphaTypes,
2035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      arraysize(kLuminanceAlphaTypes), } ,
2045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
2055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (size_t ii = 0; ii < arraysize(kFormatTypes); ++ii) {
2065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const FormatInfo& info = kFormatTypes[ii];
2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    ValueValidator<GLenum>& validator = texture_format_validators_[info.format];
2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    for (size_t jj = 0; jj < info.count; ++jj) {
2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      validator.AddValue(info.types[jj]);
2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
2115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
21458537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)bool FeatureInfo::Initialize() {
2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  disallowed_features_ = DisallowedFeatures();
21658537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  InitializeFeatures();
2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
22058537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)bool FeatureInfo::Initialize(const DisallowedFeatures& disallowed_features) {
2215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  disallowed_features_ = disallowed_features;
22258537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  InitializeFeatures();
2235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return true;
2245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
22658537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)void FeatureInfo::InitializeFeatures() {
2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Figure out what extensions to turn on.
2282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  StringSet extensions(
2292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      reinterpret_cast<const char*>(glGetString(GL_EXTENSIONS)));
2305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
231f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  const char* renderer_str =
232f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      reinterpret_cast<const char*>(glGetString(GL_RENDERER));
233f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  if (renderer_str) {
234f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    feature_flags_.is_angle = StartsWithASCII(renderer_str, "ANGLE", true);
235f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  }
236f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)
237f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  bool is_es3 = false;
238f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  const char* version_str =
239f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      reinterpret_cast<const char*>(glGetString(GL_VERSION));
240f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  if (version_str) {
2416e8cce623b6e4fe0c9e4af605d675dd9d0338c38Torne (Richard Coles)    std::string lstr(base::StringToLowerASCII(std::string(version_str)));
242f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    is_es3 = (lstr.substr(0, 12) == "opengl es 3.");
243f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  }
244f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)
2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_ANGLE_translated_shader_source");
2462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  AddExtensionString("GL_CHROMIUM_async_pixel_transfers");
2475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_bind_uniform_location");
2485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_command_buffer_query");
2495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_command_buffer_latency_query");
2505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_copy_texture");
2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_get_error_query");
2522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  AddExtensionString("GL_CHROMIUM_lose_context");
2532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  AddExtensionString("GL_CHROMIUM_pixel_transfer_buffer_object");
2545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_rate_limit_offscreen_context");
2555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_resize");
2565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_resource_safe");
2575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_strict_attribs");
2585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_CHROMIUM_texture_mailbox");
2595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  AddExtensionString("GL_EXT_debug_marker");
2605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // OES_vertex_array_object is emulated if not present natively,
2622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // so the extension string is always exposed.
2632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  AddExtensionString("GL_OES_vertex_array_object");
2642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!disallowed_features_.gpu_memory_manager)
2665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_CHROMIUM_gpu_memory_manager");
2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_ANGLE_translated_shader_source")) {
2695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.angle_translated_shader_source = true;
2705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check if we should allow GL_EXT_texture_compression_dxt1 and
2735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // GL_EXT_texture_compression_s3tc.
2745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_dxt1 = false;
2755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_dxt3 = false;
2765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_dxt5 = false;
2772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool have_s3tc = extensions.Contains("GL_EXT_texture_compression_s3tc");
2782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool have_dxt3 =
2792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      have_s3tc || extensions.Contains("GL_ANGLE_texture_compression_dxt3");
2802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool have_dxt5 =
2812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      have_s3tc || extensions.Contains("GL_ANGLE_texture_compression_dxt5");
2825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_EXT_texture_compression_dxt1") || have_s3tc) {
2845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_dxt1 = true;
2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (have_dxt3) {
2875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_dxt3 = true;
2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (have_dxt5) {
2905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_dxt5 = true;
2915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_dxt1) {
2945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_EXT_texture_compression_dxt1");
2955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.compressed_texture_format.AddValue(
2965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        GL_COMPRESSED_RGB_S3TC_DXT1_EXT);
2975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.compressed_texture_format.AddValue(
2985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        GL_COMPRESSED_RGBA_S3TC_DXT1_EXT);
2995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_dxt3) {
3025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // The difference between GL_EXT_texture_compression_s3tc and
3035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // GL_CHROMIUM_texture_compression_dxt3 is that the former
3045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // requires on the fly compression. The latter does not.
3055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_CHROMIUM_texture_compression_dxt3");
3065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.compressed_texture_format.AddValue(
3075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        GL_COMPRESSED_RGBA_S3TC_DXT3_EXT);
3085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_dxt5) {
3115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // The difference between GL_EXT_texture_compression_s3tc and
3125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // GL_CHROMIUM_texture_compression_dxt5 is that the former
3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // requires on the fly compression. The latter does not.
3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_CHROMIUM_texture_compression_dxt5");
3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.compressed_texture_format.AddValue(
3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        GL_COMPRESSED_RGBA_S3TC_DXT5_EXT);
3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check if we should enable GL_EXT_texture_filter_anisotropic.
3202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_EXT_texture_filter_anisotropic")) {
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_EXT_texture_filter_anisotropic");
3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_parameter.AddValue(
3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        GL_TEXTURE_MAX_ANISOTROPY_EXT);
3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.g_l_state.AddValue(
3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT);
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check if we should support GL_OES_packed_depth_stencil and/or
3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // GL_GOOGLE_depth_texture / GL_CHROMIUM_depth_texture.
3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
3315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // NOTE: GL_OES_depth_texture requires support for depth cubemaps.
3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // GL_ARB_depth_texture requires other features that
3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // GL_OES_packed_depth_stencil does not provide.
3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Therefore we made up GL_GOOGLE_depth_texture / GL_CHROMIUM_depth_texture.
3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // GL_GOOGLE_depth_texture is legacy. As we exposed it into NaCl we can't
3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // get rid of it.
3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_depth_texture = false;
34190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (!workarounds_.disable_depth_texture &&
342c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      (extensions.Contains("GL_ARB_depth_texture") ||
343c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)       extensions.Contains("GL_OES_depth_texture") ||
344f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)       extensions.Contains("GL_ANGLE_depth_texture") || is_es3)) {
3455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_depth_texture = true;
346f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    feature_flags_.angle_depth_texture =
347f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)        extensions.Contains("GL_ANGLE_depth_texture");
3485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_depth_texture) {
3515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_CHROMIUM_depth_texture");
3525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_GOOGLE_depth_texture");
3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_DEPTH_COMPONENT].AddValue(GL_UNSIGNED_SHORT);
3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_DEPTH_COMPONENT].AddValue(GL_UNSIGNED_INT);
3555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_internal_format.AddValue(GL_DEPTH_COMPONENT);
3565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_format.AddValue(GL_DEPTH_COMPONENT);
3575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.pixel_type.AddValue(GL_UNSIGNED_SHORT);
3585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.pixel_type.AddValue(GL_UNSIGNED_INT);
3595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_EXT_packed_depth_stencil") ||
362f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      extensions.Contains("GL_OES_packed_depth_stencil") || is_es3) {
3635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_packed_depth_stencil");
364f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    feature_flags_.packed_depth24_stencil8 = true;
3655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (enable_depth_texture) {
366f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      texture_format_validators_[GL_DEPTH_STENCIL]
367f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)          .AddValue(GL_UNSIGNED_INT_24_8);
3685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      validators_.texture_internal_format.AddValue(GL_DEPTH_STENCIL);
3695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      validators_.texture_format.AddValue(GL_DEPTH_STENCIL);
3705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      validators_.pixel_type.AddValue(GL_UNSIGNED_INT_24_8);
3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
3725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.render_buffer_format.AddValue(GL_DEPTH24_STENCIL8);
3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3751320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  if (is_es3 || extensions.Contains("GL_OES_vertex_array_object") ||
3762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_ARB_vertex_array_object") ||
3772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_APPLE_vertex_array_object")) {
3782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    feature_flags_.native_vertex_array_object = true;
3792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
3805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // If we're using client_side_arrays we have to emulate
3822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // vertex array objects since vertex array objects do not work
3832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // with client side arrays.
3842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (workarounds_.use_client_side_arrays_for_stream_buffers) {
3852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    feature_flags_.native_vertex_array_object = false;
3862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
3872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3881320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  if (is_es3 || extensions.Contains("GL_OES_element_index_uint") ||
3892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      gfx::HasDesktopGLFeatures()) {
3902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    AddExtensionString("GL_OES_element_index_uint");
3912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    validators_.index_type.AddValue(GL_UNSIGNED_INT);
3925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_texture_format_bgra8888 = false;
3955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_read_format_bgra = false;
3968bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)  bool enable_render_buffer_bgra = false;
3971320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  bool enable_immutable_texture_format_bgra_on_es3 =
3981320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      extensions.Contains("GL_APPLE_texture_format_BGRA8888");
3998bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)
4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check if we should allow GL_EXT_texture_format_BGRA8888
4012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_EXT_texture_format_BGRA8888") ||
4021320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      enable_immutable_texture_format_bgra_on_es3 ||
4032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_EXT_bgra")) {
4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_texture_format_bgra8888 = true;
4055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_EXT_bgra")) {
4088bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)    enable_render_buffer_bgra = true;
4095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_EXT_read_format_bgra") ||
4122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_EXT_bgra")) {
4135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_read_format_bgra = true;
4145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_texture_format_bgra8888) {
417a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    feature_flags_.ext_texture_format_bgra8888 = true;
4185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_EXT_texture_format_BGRA8888");
4195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_BGRA_EXT].AddValue(GL_UNSIGNED_BYTE);
4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_internal_format.AddValue(GL_BGRA_EXT);
4215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_format.AddValue(GL_BGRA_EXT);
4225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_read_format_bgra) {
4255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_EXT_read_format_bgra");
4265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.read_pixel_format.AddValue(GL_BGRA_EXT);
4275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4298bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)  if (enable_render_buffer_bgra) {
4308bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)    AddExtensionString("GL_CHROMIUM_renderbuffer_format_BGRA8888");
4318bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)    validators_.render_buffer_format.AddValue(GL_BGRA8_EXT);
4328bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)  }
4338bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)
4342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_OES_rgb8_rgba8") || gfx::HasDesktopGLFeatures()) {
4352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    AddExtensionString("GL_OES_rgb8_rgba8");
4362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    validators_.render_buffer_format.AddValue(GL_RGB8_OES);
4372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    validators_.render_buffer_format.AddValue(GL_RGBA8_OES);
4385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check if we should allow GL_OES_texture_npot
4411320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  if (is_es3 || extensions.Contains("GL_ARB_texture_non_power_of_two") ||
4422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_OES_texture_npot")) {
4435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_texture_npot");
4441320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    feature_flags_.npot_ok = true;
4455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check if we should allow GL_OES_texture_float, GL_OES_texture_half_float,
4485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // GL_OES_texture_float_linear, GL_OES_texture_half_float_linear
4495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_texture_float = false;
4505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_texture_float_linear = false;
4515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_texture_half_float = false;
4525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool enable_texture_half_float_linear = false;
4535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  bool may_enable_chromium_color_buffer_float = false;
4555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (extensions.Contains("GL_ARB_texture_float")) {
4575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_texture_float = true;
4585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_texture_float_linear = true;
4595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_texture_half_float = true;
4605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    enable_texture_half_float_linear = true;
4615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    may_enable_chromium_color_buffer_float = true;
4625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
4631320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    if (is_es3 || extensions.Contains("GL_OES_texture_float")) {
4645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      enable_texture_float = true;
4655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if (extensions.Contains("GL_OES_texture_float_linear")) {
4665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        enable_texture_float_linear = true;
4675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
4685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if ((is_es3 && extensions.Contains("GL_EXT_color_buffer_float")) ||
4695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)          feature_flags_.is_angle) {
4705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        may_enable_chromium_color_buffer_float = true;
4715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      }
4725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4731320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    // TODO(dshwang): GLES3 supports half float by default but GL_HALF_FLOAT_OES
4741320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    // isn't equal to GL_HALF_FLOAT.
4755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (extensions.Contains("GL_OES_texture_half_float")) {
4765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      enable_texture_half_float = true;
4775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if (extensions.Contains("GL_OES_texture_half_float_linear")) {
4785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        enable_texture_half_float_linear = true;
4795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      }
4805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_texture_float) {
4845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_ALPHA].AddValue(GL_FLOAT);
4855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_RGB].AddValue(GL_FLOAT);
4865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_RGBA].AddValue(GL_FLOAT);
4875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_LUMINANCE].AddValue(GL_FLOAT);
4885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_LUMINANCE_ALPHA].AddValue(GL_FLOAT);
4895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.pixel_type.AddValue(GL_FLOAT);
4905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.read_pixel_type.AddValue(GL_FLOAT);
4915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_texture_float");
4925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (enable_texture_float_linear) {
4935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      AddExtensionString("GL_OES_texture_float_linear");
4945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
4955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (enable_texture_half_float) {
4985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_ALPHA].AddValue(GL_HALF_FLOAT_OES);
4995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_RGB].AddValue(GL_HALF_FLOAT_OES);
5005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_RGBA].AddValue(GL_HALF_FLOAT_OES);
5015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_LUMINANCE].AddValue(GL_HALF_FLOAT_OES);
5025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    texture_format_validators_[GL_LUMINANCE_ALPHA].AddValue(GL_HALF_FLOAT_OES);
5035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.pixel_type.AddValue(GL_HALF_FLOAT_OES);
5045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.read_pixel_type.AddValue(GL_HALF_FLOAT_OES);
5055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_texture_half_float");
5065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (enable_texture_half_float_linear) {
5075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      AddExtensionString("GL_OES_texture_half_float_linear");
5085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
5095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
5105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5115d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (may_enable_chromium_color_buffer_float) {
5125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    COMPILE_ASSERT(GL_RGBA32F_ARB == GL_RGBA32F &&
5135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                   GL_RGBA32F_EXT == GL_RGBA32F &&
5145d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                   GL_RGB32F_ARB == GL_RGB32F &&
5155d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                   GL_RGB32F_EXT == GL_RGB32F,
5165d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                   sized_float_internal_format_variations_must_match);
5175d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // We don't check extension support beyond ARB_texture_float on desktop GL,
5185d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // and format support varies between GL configurations. For example, spec
5195d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // prior to OpenGL 3.0 mandates framebuffer support only for one
5205d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // implementation-chosen format, and ES3.0 EXT_color_buffer_float does not
5215d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // support rendering to RGB32F. Check for framebuffer completeness with
5225d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // formats that the extensions expose, and only enable an extension when a
5235d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // framebuffer created with its texture format is reported as complete.
5245d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLint fb_binding = 0;
5255d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLint tex_binding = 0;
5265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glGetIntegerv(GL_FRAMEBUFFER_BINDING, &fb_binding);
5275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glGetIntegerv(GL_TEXTURE_BINDING_2D, &tex_binding);
5285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
5295d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLuint tex_id = 0;
5305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLuint fb_id = 0;
5315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLsizei width = 16;
5325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
5335d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glGenTextures(1, &tex_id);
5345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glGenFramebuffersEXT(1, &fb_id);
5355d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glBindTexture(GL_TEXTURE_2D, tex_id);
5365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // Nearest filter needed for framebuffer completeness on some drivers.
5375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
5385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA32F, width, width, 0, GL_RGBA,
5395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                 GL_FLOAT, NULL);
5405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glBindFramebufferEXT(GL_FRAMEBUFFER, fb_id);
5415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glFramebufferTexture2DEXT(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
5425d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                              GL_TEXTURE_2D, tex_id, 0);
5435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLenum statusRGBA = glCheckFramebufferStatusEXT(GL_FRAMEBUFFER);
5445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB32F, width, width, 0, GL_RGB,
5455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                 GL_FLOAT, NULL);
5465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GLenum statusRGB = glCheckFramebufferStatusEXT(GL_FRAMEBUFFER);
5475d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glDeleteFramebuffersEXT(1, &fb_id);
5485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glDeleteTextures(1, &tex_id);
5495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
5505d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glBindFramebufferEXT(GL_FRAMEBUFFER, static_cast<GLuint>(fb_binding));
5515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    glBindTexture(GL_TEXTURE_2D, static_cast<GLuint>(tex_binding));
5525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
5535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    DCHECK(glGetError() == GL_NO_ERROR);
5545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
5555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (statusRGBA == GL_FRAMEBUFFER_COMPLETE) {
5565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      validators_.texture_internal_format.AddValue(GL_RGBA32F);
5575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      feature_flags_.chromium_color_buffer_float_rgba = true;
5585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      AddExtensionString("GL_CHROMIUM_color_buffer_float_rgba");
5595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
5605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (statusRGB == GL_FRAMEBUFFER_COMPLETE) {
5615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      validators_.texture_internal_format.AddValue(GL_RGB32F);
5625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      feature_flags_.chromium_color_buffer_float_rgb = true;
5635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      AddExtensionString("GL_CHROMIUM_color_buffer_float_rgb");
5645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
5655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
5665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
5675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Check for multisample support
5685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (!workarounds_.disable_multisampling) {
5697dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    bool ext_has_multisample =
570f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)        extensions.Contains("GL_EXT_framebuffer_multisample") || is_es3;
571f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    if (feature_flags_.is_angle) {
5727dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      ext_has_multisample |=
573f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)          extensions.Contains("GL_ANGLE_framebuffer_multisample");
5747dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    }
575f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    feature_flags_.use_core_framebuffer_multisample = is_es3;
5767dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    if (ext_has_multisample) {
5777dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      feature_flags_.chromium_framebuffer_multisample = true;
5787dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      validators_.frame_buffer_target.AddValue(GL_READ_FRAMEBUFFER_EXT);
5797dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      validators_.frame_buffer_target.AddValue(GL_DRAW_FRAMEBUFFER_EXT);
5807dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      validators_.g_l_state.AddValue(GL_READ_FRAMEBUFFER_BINDING_EXT);
5817dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      validators_.g_l_state.AddValue(GL_MAX_SAMPLES_EXT);
5827dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      validators_.render_buffer_parameter.AddValue(GL_RENDERBUFFER_SAMPLES_EXT);
5837dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      AddExtensionString("GL_CHROMIUM_framebuffer_multisample");
5845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
5855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (extensions.Contains("GL_EXT_multisampled_render_to_texture")) {
5865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      feature_flags_.multisampled_render_to_texture = true;
5875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    } else if (extensions.Contains("GL_IMG_multisampled_render_to_texture")) {
5885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      feature_flags_.multisampled_render_to_texture = true;
5895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      feature_flags_.use_img_for_multisampled_render_to_texture = true;
5905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
5915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (feature_flags_.multisampled_render_to_texture) {
5925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      validators_.render_buffer_parameter.AddValue(
5935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)          GL_RENDERBUFFER_SAMPLES_EXT);
5945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      validators_.g_l_state.AddValue(GL_MAX_SAMPLES_EXT);
5955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      validators_.frame_buffer_parameter.AddValue(
5965d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)          GL_FRAMEBUFFER_ATTACHMENT_TEXTURE_SAMPLES_EXT);
5975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      AddExtensionString("GL_EXT_multisampled_render_to_texture");
5987dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    }
5995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
601f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  if (extensions.Contains("GL_OES_depth24") || gfx::HasDesktopGLFeatures() ||
602f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      is_es3) {
6035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_depth24");
60468043e1e95eeb07d5cae7aca370b26518b0867d6Torne (Richard Coles)    feature_flags_.oes_depth24 = true;
6055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.render_buffer_format.AddValue(GL_DEPTH_COMPONENT24);
6065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
60890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (!workarounds_.disable_oes_standard_derivatives &&
6091320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      (is_es3 || extensions.Contains("GL_OES_standard_derivatives") ||
61090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)       gfx::HasDesktopGLFeatures())) {
6115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_standard_derivatives");
6125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.oes_standard_derivatives = true;
6135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.hint_target.AddValue(GL_FRAGMENT_SHADER_DERIVATIVE_HINT_OES);
6145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.g_l_state.AddValue(GL_FRAGMENT_SHADER_DERIVATIVE_HINT_OES);
6155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_OES_EGL_image_external")) {
6185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_EGL_image_external");
6195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.oes_egl_image_external = true;
6205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_bind_target.AddValue(GL_TEXTURE_EXTERNAL_OES);
6215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.get_tex_param_target.AddValue(GL_TEXTURE_EXTERNAL_OES);
6225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_parameter.AddValue(GL_REQUIRED_TEXTURE_IMAGE_UNITS_OES);
6235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.g_l_state.AddValue(GL_TEXTURE_BINDING_EXTERNAL_OES);
6245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_OES_compressed_ETC1_RGB8_texture")) {
6275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_OES_compressed_ETC1_RGB8_texture");
628a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    feature_flags_.oes_compressed_etc1_rgb8_texture = true;
6295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.compressed_texture_format.AddValue(GL_ETC1_RGB8_OES);
6305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6325c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (extensions.Contains("GL_AMD_compressed_ATC_texture")) {
6335c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    AddExtensionString("GL_AMD_compressed_ATC_texture");
6345c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6355c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_ATC_RGB_AMD);
6365c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6375c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_ATC_RGBA_EXPLICIT_ALPHA_AMD);
6385c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6395c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_ATC_RGBA_INTERPOLATED_ALPHA_AMD);
6405c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  }
6415c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu
6425c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  if (extensions.Contains("GL_IMG_texture_compression_pvrtc")) {
6435c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    AddExtensionString("GL_IMG_texture_compression_pvrtc");
6445c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6455c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG);
6465c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6475c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG);
6485c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6495c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG);
6505c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    validators_.compressed_texture_format.AddValue(
6515c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG);
6525c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  }
6535c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu
6545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Ideally we would only expose this extension on Mac OS X, to
6555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // support GL_CHROMIUM_iosurface and the compositor. We don't want
6565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // applications to start using it; they should use ordinary non-
6575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // power-of-two textures. However, for unit testing purposes we
6585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // expose it on all supported platforms.
6592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_ARB_texture_rectangle")) {
6605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_ARB_texture_rectangle");
6615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.arb_texture_rectangle = true;
6625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_bind_target.AddValue(GL_TEXTURE_RECTANGLE_ARB);
6635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // For the moment we don't add this enum to the texture_target
6645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // validator. This implies that the only way to get image data into a
6655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // rectangular texture is via glTexImageIOSurface2DCHROMIUM, which is
6665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // just fine since again we don't want applications depending on this
6675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // extension.
6685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.get_tex_param_target.AddValue(GL_TEXTURE_RECTANGLE_ARB);
6695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.g_l_state.AddValue(GL_TEXTURE_BINDING_RECTANGLE_ARB);
6705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_MACOSX)
67346d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  AddExtensionString("GL_CHROMIUM_iosurface");
6745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
6755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TODO(gman): Add support for these extensions.
6775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //     GL_OES_depth32
6785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  feature_flags_.enable_texture_float_linear |= enable_texture_float_linear;
6805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  feature_flags_.enable_texture_half_float_linear |=
6815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      enable_texture_half_float_linear;
6825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_ANGLE_pack_reverse_row_order")) {
6845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_ANGLE_pack_reverse_row_order");
6855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.angle_pack_reverse_row_order = true;
6865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.pixel_store.AddValue(GL_PACK_REVERSE_ROW_ORDER_ANGLE);
6875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.g_l_state.AddValue(GL_PACK_REVERSE_ROW_ORDER_ANGLE);
6885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6902a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (extensions.Contains("GL_ANGLE_texture_usage")) {
691a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    feature_flags_.angle_texture_usage = true;
6925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_ANGLE_texture_usage");
6935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_parameter.AddValue(GL_TEXTURE_USAGE_ANGLE);
6945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
6955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
6961320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // Note: Only APPLE_texture_format_BGRA8888 extension allows BGRA8_EXT in
6971320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // ES3's glTexStorage2D. We prefer support BGRA to texture storage.
6981320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // So we don't expose GL_EXT_texture_storage when ES3 +
6991320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // GL_EXT_texture_format_BGRA8888 because we fail the GL_BGRA8 requirement.
7001320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // However we expose GL_EXT_texture_storage when just ES3 because we don't
7011320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // claim to handle GL_BGRA8.
7021320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  bool support_texture_storage_on_es3 =
7031320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      (is_es3 && enable_immutable_texture_format_bgra_on_es3) ||
7041320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      (is_es3 && !enable_texture_format_bgra8888);
7051320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  if (extensions.Contains("GL_EXT_texture_storage") ||
7061320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      extensions.Contains("GL_ARB_texture_storage") ||
7071320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      support_texture_storage_on_es3) {
708a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    feature_flags_.ext_texture_storage = true;
7095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_EXT_texture_storage");
7105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.texture_parameter.AddValue(GL_TEXTURE_IMMUTABLE_FORMAT_EXT);
7115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (enable_texture_format_bgra8888)
7125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_BGRA8_EXT);
7135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (enable_texture_float) {
7145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_RGBA32F_EXT);
7155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_RGB32F_EXT);
7165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_ALPHA32F_EXT);
7175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(
7185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)            GL_LUMINANCE32F_EXT);
7195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(
7205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)            GL_LUMINANCE_ALPHA32F_EXT);
7215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
7225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (enable_texture_half_float) {
7235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_RGBA16F_EXT);
7245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_RGB16F_EXT);
7255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(GL_ALPHA16F_EXT);
7265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(
7275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)            GL_LUMINANCE16F_EXT);
7285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        validators_.texture_internal_format_storage.AddValue(
7295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)            GL_LUMINANCE_ALPHA16F_EXT);
7305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
7315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
7335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool have_ext_occlusion_query_boolean =
7342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_EXT_occlusion_query_boolean");
7352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool have_arb_occlusion_query2 =
7362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_ARB_occlusion_query2");
7372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  bool have_arb_occlusion_query =
7382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      extensions.Contains("GL_ARB_occlusion_query");
7395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
740c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (!workarounds_.disable_ext_occlusion_query &&
7415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      (have_ext_occlusion_query_boolean ||
7425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       have_arb_occlusion_query2 ||
7435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       have_arb_occlusion_query)) {
7445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_EXT_occlusion_query_boolean");
7455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.occlusion_query_boolean = true;
7465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.use_arb_occlusion_query2_for_occlusion_query_boolean =
7475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        !have_ext_occlusion_query_boolean && have_arb_occlusion_query2;
7485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.use_arb_occlusion_query_for_occlusion_query_boolean =
7495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        !have_ext_occlusion_query_boolean && have_arb_occlusion_query &&
7505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        !have_arb_occlusion_query2;
7515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
753424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  if (!workarounds_.disable_angle_instanced_arrays &&
754424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)      (extensions.Contains("GL_ANGLE_instanced_arrays") ||
755424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)       (extensions.Contains("GL_ARB_instanced_arrays") &&
756a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch        extensions.Contains("GL_ARB_draw_instanced")) ||
757a02191e04bc25c4935f804f2c080ae28663d096dBen Murdoch       is_es3)) {
7585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    AddExtensionString("GL_ANGLE_instanced_arrays");
7595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    feature_flags_.angle_instanced_arrays = true;
7605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    validators_.vertex_attribute.AddValue(GL_VERTEX_ATTRIB_ARRAY_DIVISOR_ANGLE);
7615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
7625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
763c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (!workarounds_.disable_ext_draw_buffers &&
764c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      (extensions.Contains("GL_ARB_draw_buffers") ||
765c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)       extensions.Contains("GL_EXT_draw_buffers"))) {
7662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    AddExtensionString("GL_EXT_draw_buffers");
7672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    feature_flags_.ext_draw_buffers = true;
7682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
7692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    GLint max_color_attachments = 0;
7702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    glGetIntegerv(GL_MAX_COLOR_ATTACHMENTS_EXT, &max_color_attachments);
7712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    for (GLenum i = GL_COLOR_ATTACHMENT1_EXT;
7722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)         i < static_cast<GLenum>(GL_COLOR_ATTACHMENT0 + max_color_attachments);
7732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)         ++i) {
7742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      validators_.attachment.AddValue(i);
7752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    }
7761320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    COMPILE_ASSERT(GL_COLOR_ATTACHMENT0_EXT == GL_COLOR_ATTACHMENT0,
7771320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci                   color_attachment0_variation_must_match);
7782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
7792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    validators_.g_l_state.AddValue(GL_MAX_COLOR_ATTACHMENTS_EXT);
7802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    validators_.g_l_state.AddValue(GL_MAX_DRAW_BUFFERS_ARB);
7812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    GLint max_draw_buffers = 0;
7822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    glGetIntegerv(GL_MAX_DRAW_BUFFERS_ARB, &max_draw_buffers);
7832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    for (GLenum i = GL_DRAW_BUFFER0_ARB;
7842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)         i < static_cast<GLenum>(GL_DRAW_BUFFER0_ARB + max_draw_buffers);
7852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)         ++i) {
7862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      validators_.g_l_state.AddValue(i);
7872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    }
7882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
7892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
7901320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  if (is_es3 || extensions.Contains("GL_EXT_blend_minmax") ||
791cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)      gfx::HasDesktopGLFeatures()) {
792cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    AddExtensionString("GL_EXT_blend_minmax");
793cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    validators_.equation.AddValue(GL_MIN_EXT);
794cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    validators_.equation.AddValue(GL_MAX_EXT);
7951320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    COMPILE_ASSERT(GL_MIN_EXT == GL_MIN && GL_MAX_EXT == GL_MAX,
7961320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci                   min_max_variations_must_match);
797cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  }
798cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)
7991320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  // TODO(dshwang): GLES3 supports gl_FragDepth, not gl_FragDepthEXT.
800868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  if (extensions.Contains("GL_EXT_frag_depth") || gfx::HasDesktopGLFeatures()) {
801868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    AddExtensionString("GL_EXT_frag_depth");
802868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    feature_flags_.ext_frag_depth = true;
803868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  }
804868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
8050529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  if (extensions.Contains("GL_EXT_shader_texture_lod") ||
8060529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      gfx::HasDesktopGLFeatures()) {
8070529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    AddExtensionString("GL_EXT_shader_texture_lod");
8080529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    feature_flags_.ext_shader_texture_lod = true;
8090529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  }
8100529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch
811cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#if !defined(OS_MACOSX)
812cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  if (workarounds_.disable_egl_khr_fence_sync) {
813cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)    gfx::g_driver_egl.ext.b_EGL_KHR_fence_sync = false;
814cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)  }
8158b70b8257c44a011969316b68be9b5441786809eboliu  if (workarounds_.disable_egl_khr_wait_sync) {
8168b70b8257c44a011969316b68be9b5441786809eboliu    gfx::g_driver_egl.ext.b_EGL_KHR_wait_sync = false;
8178b70b8257c44a011969316b68be9b5441786809eboliu  }
818cedac228d2dd51db4b79ea1e72c7f249408ee061Torne (Richard Coles)#endif
81946d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)  if (workarounds_.disable_arb_sync)
82046d4c2bc3267f3f028f39e7e311b0f89aba2e4fdTorne (Richard Coles)    gfx::g_driver_gl.ext.b_GL_ARB_sync = false;
821116680a4aac90f2aa7413d9095a592090648e557Ben Murdoch  bool ui_gl_fence_works = gfx::GLFence::IsSupported();
8220529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  UMA_HISTOGRAM_BOOLEAN("GPU.FenceSupport", ui_gl_fence_works);
823424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)
824424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  feature_flags_.map_buffer_range =
825424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)      is_es3 || extensions.Contains("GL_ARB_map_buffer_range");
826424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)
827424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  // Really it's part of core OpenGL 2.1 and up, but let's assume the
828424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  // extension is still advertised.
829424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  bool has_pixel_buffers =
830424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)      is_es3 || extensions.Contains("GL_ARB_pixel_buffer_object");
831424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)
832424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  // We will use either glMapBuffer() or glMapBufferRange() for async readbacks.
833424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  if (has_pixel_buffers && ui_gl_fence_works &&
834424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)      !workarounds_.disable_async_readpixels) {
835424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)    feature_flags_.use_async_readpixels = true;
836424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  }
837424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)
8382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (is_es3 || extensions.Contains("GL_ARB_sampler_objects")) {
8392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    feature_flags_.enable_samplers = true;
8402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    // TODO(dsinclair): Add AddExtensionString("GL_CHROMIUM_sampler_objects")
8412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    // when available.
8422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
84358537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)
844d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles)  if ((is_es3 || extensions.Contains("GL_EXT_discard_framebuffer")) &&
845d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles)      !workarounds_.disable_ext_discard_framebuffer) {
84658537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    // DiscardFramebufferEXT is automatically bound to InvalidateFramebuffer.
84758537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    AddExtensionString("GL_EXT_discard_framebuffer");
8480f1bc08d4cfcc34181b0b5cbf065c40f687bf740Torne (Richard Coles)    feature_flags_.ext_discard_framebuffer = true;
84958537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  }
8500529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch
8510529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  if (ui_gl_fence_works) {
8520529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    AddExtensionString("GL_CHROMIUM_sync_query");
8530529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch    feature_flags_.chromium_sync_query = true;
8540529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  }
8551320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci
8561320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  if (extensions.Contains("GL_NV_path_rendering")) {
8571320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    if (extensions.Contains("GL_EXT_direct_state_access") || is_es3) {
8581320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      AddExtensionString("GL_CHROMIUM_path_rendering");
8591320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      feature_flags_.chromium_path_rendering = true;
8601320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      validators_.g_l_state.AddValue(GL_PATH_MODELVIEW_MATRIX_CHROMIUM);
8611320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      validators_.g_l_state.AddValue(GL_PATH_PROJECTION_MATRIX_CHROMIUM);
8621320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    }
8631320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  }
8645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8661320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tuccivoid FeatureInfo::AddExtensionString(const char* s) {
8671320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  std::string str(s);
8685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  size_t pos = extensions_.find(str);
8695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  while (pos != std::string::npos &&
8705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)         pos + str.length() < extensions_.length() &&
8715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)         extensions_.substr(pos + str.length(), 1) != " ") {
8725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // This extension name is a substring of another.
8735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    pos = extensions_.find(str, pos + str.length());
8745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
8755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (pos == std::string::npos) {
8765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    extensions_ += (extensions_.empty() ? "" : " ") + str;
8775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
8785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)FeatureInfo::~FeatureInfo() {
8815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
8825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
8835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace gles2
8845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace gpu
885