| // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef GPU_COMMAND_BUFFER_CLIENT_GLES2_IMPLEMENTATION_H_ |
| #define GPU_COMMAND_BUFFER_CLIENT_GLES2_IMPLEMENTATION_H_ |
| |
| #include <stddef.h> |
| #include <stdint.h> |
| |
| #include <list> |
| #include <map> |
| #include <memory> |
| #include <set> |
| #include <string> |
| #include <unordered_map> |
| #include <utility> |
| #include <vector> |
| |
| #include "base/compiler_specific.h" |
| #include "base/containers/queue.h" |
| #include "base/macros.h" |
| #include "base/memory/weak_ptr.h" |
| #include "base/optional.h" |
| #include "base/trace_event/memory_dump_provider.h" |
| #include "gpu/command_buffer/client/buffer_tracker.h" |
| #include "gpu/command_buffer/client/client_context_state.h" |
| #include "gpu/command_buffer/client/client_transfer_cache.h" |
| #include "gpu/command_buffer/client/context_support.h" |
| #include "gpu/command_buffer/client/gles2_impl_export.h" |
| #include "gpu/command_buffer/client/gles2_interface.h" |
| #include "gpu/command_buffer/client/gpu_control_client.h" |
| #include "gpu/command_buffer/client/implementation_base.h" |
| #include "gpu/command_buffer/client/logging.h" |
| #include "gpu/command_buffer/client/mapped_memory.h" |
| #include "gpu/command_buffer/client/query_tracker.h" |
| #include "gpu/command_buffer/client/readback_buffer_shadow_tracker.h" |
| #include "gpu/command_buffer/client/ref_counted.h" |
| #include "gpu/command_buffer/client/share_group.h" |
| #include "gpu/command_buffer/client/transfer_buffer.h" |
| #include "gpu/command_buffer/common/context_creation_attribs.h" |
| #include "gpu/command_buffer/common/context_result.h" |
| #include "gpu/command_buffer/common/debug_marker_manager.h" |
| |
| namespace gpu { |
| |
| class IdAllocator; |
| |
| namespace gles2 { |
| |
| class GLES2CmdHelper; |
| class VertexArrayObjectManager; |
| class ReadbackBufferShadowTracker; |
| |
| // This class emulates GLES2 over command buffers. It can be used by a client |
| // program so that the program does not need deal with shared memory and command |
| // buffer management. See gl2_lib.h. Note that there is a performance gain to |
| // be had by changing your code to use command buffers directly by using the |
| // GLES2CmdHelper but that entails changing your code to use and deal with |
| // shared memory and synchronization issues. |
| class GLES2_IMPL_EXPORT GLES2Implementation : public GLES2Interface, |
| public ImplementationBase, |
| public QueryTrackerClient { |
| public: |
| // Stores GL state that never changes. |
| struct GLES2_IMPL_EXPORT GLStaticState { |
| GLStaticState(); |
| ~GLStaticState(); |
| |
| typedef std::pair<GLenum, GLenum> ShaderPrecisionKey; |
| typedef std::map<ShaderPrecisionKey, |
| cmds::GetShaderPrecisionFormat::Result> |
| ShaderPrecisionMap; |
| ShaderPrecisionMap shader_precisions; |
| }; |
| |
| // GL names for the buffers used to emulate client side buffers. |
| static const GLuint kClientSideArrayId = 0xFEDCBA98u; |
| static const GLuint kClientSideElementArrayId = 0xFEDCBA99u; |
| |
| // Number of swap buffers allowed before waiting. |
| static const size_t kMaxSwapBuffers = 2; |
| |
| GLES2Implementation(GLES2CmdHelper* helper, |
| scoped_refptr<ShareGroup> share_group, |
| TransferBufferInterface* transfer_buffer, |
| bool bind_generates_resource, |
| bool lose_context_when_out_of_memory, |
| bool support_client_side_arrays, |
| GpuControl* gpu_control); |
| |
| ~GLES2Implementation() override; |
| |
| gpu::ContextResult Initialize(const SharedMemoryLimits& limits); |
| |
| // The GLES2CmdHelper being used by this GLES2Implementation. You can use |
| // this to issue cmds at a lower level for certain kinds of optimization. |
| GLES2CmdHelper* helper() const; |
| |
| // GLES2Interface implementation |
| void FreeSharedMemory(void*) override; |
| |
| // Include the auto-generated part of this class. We split this because |
| // it means we can easily edit the non-auto generated parts right here in |
| // this file instead of having to edit some template or the code generator. |
| #include "gpu/command_buffer/client/gles2_implementation_autogen.h" |
| |
| // ContextSupport implementation. |
| void SetAggressivelyFreeResources(bool aggressively_free_resources) override; |
| void Swap(uint32_t flags, |
| SwapCompletedCallback complete_callback, |
| PresentationCallback presentation_callback) override; |
| void SwapWithBounds(const std::vector<gfx::Rect>& rects, |
| uint32_t flags, |
| SwapCompletedCallback swap_completed, |
| PresentationCallback presentation_callback) override; |
| void PartialSwapBuffers(const gfx::Rect& sub_buffer, |
| uint32_t flags, |
| SwapCompletedCallback swap_completed, |
| PresentationCallback presentation_callback) override; |
| void CommitOverlayPlanes(uint32_t flags, |
| SwapCompletedCallback swap_completed, |
| PresentationCallback presentation_callback) override; |
| void ScheduleOverlayPlane(int plane_z_order, |
| gfx::OverlayTransform plane_transform, |
| unsigned overlay_texture_id, |
| const gfx::Rect& display_bounds, |
| const gfx::RectF& uv_rect, |
| bool enable_blend, |
| unsigned gpu_fence_id) override; |
| uint64_t ShareGroupTracingGUID() const override; |
| void SetErrorMessageCallback( |
| base::RepeatingCallback<void(const char*, int32_t)> callback) override; |
| bool ThreadSafeShallowLockDiscardableTexture(uint32_t texture_id) override; |
| void CompleteLockDiscardableTexureOnContextThread( |
| uint32_t texture_id) override; |
| bool ThreadsafeDiscardableTextureIsDeletedForTracing( |
| uint32_t texture_id) override; |
| void* MapTransferCacheEntry(uint32_t serialized_size) override; |
| void UnmapAndCreateTransferCacheEntry(uint32_t type, uint32_t id) override; |
| bool ThreadsafeLockTransferCacheEntry(uint32_t type, uint32_t id) override; |
| void UnlockTransferCacheEntries( |
| const std::vector<std::pair<uint32_t, uint32_t>>& entries) override; |
| void DeleteTransferCacheEntry(uint32_t type, uint32_t id) override; |
| unsigned int GetTransferBufferFreeSize() const override; |
| bool CanDecodeWithHardwareAcceleration( |
| base::span<const uint8_t> encoded_data) const override; |
| |
| // InterfaceBase implementation. |
| void GenSyncTokenCHROMIUM(GLbyte* sync_token) override; |
| void GenUnverifiedSyncTokenCHROMIUM(GLbyte* sync_token) override; |
| void VerifySyncTokensCHROMIUM(GLbyte** sync_tokens, GLsizei count) override; |
| void WaitSyncTokenCHROMIUM(const GLbyte* sync_token) override; |
| |
| void GetProgramInfoCHROMIUMHelper(GLuint program, |
| std::vector<int8_t>* result); |
| GLint GetAttribLocationHelper(GLuint program, const char* name); |
| GLint GetUniformLocationHelper(GLuint program, const char* name); |
| GLint GetFragDataIndexEXTHelper(GLuint program, const char* name); |
| GLint GetFragDataLocationHelper(GLuint program, const char* name); |
| |
| // Writes the result bucket into a buffer pointed by name and of maximum size |
| // buffsize. If length is !null, it receives the number of characters written |
| // (excluding the final \0). This is a helper function for GetActive*Helper |
| // functions that return names. |
| void GetResultNameHelper(GLsizei bufsize, GLsizei* length, char* name); |
| bool GetActiveAttribHelper( |
| GLuint program, GLuint index, GLsizei bufsize, GLsizei* length, |
| GLint* size, GLenum* type, char* name); |
| bool GetActiveUniformHelper( |
| GLuint program, GLuint index, GLsizei bufsize, GLsizei* length, |
| GLint* size, GLenum* type, char* name); |
| void GetUniformBlocksCHROMIUMHelper(GLuint program, |
| std::vector<int8_t>* result); |
| void GetUniformsES3CHROMIUMHelper(GLuint program, |
| std::vector<int8_t>* result); |
| GLuint GetUniformBlockIndexHelper(GLuint program, const char* name); |
| bool GetActiveUniformBlockNameHelper( |
| GLuint program, GLuint index, GLsizei bufsize, |
| GLsizei* length, char* name); |
| bool GetActiveUniformBlockivHelper( |
| GLuint program, GLuint index, GLenum pname, GLint* params); |
| void GetTransformFeedbackVaryingsCHROMIUMHelper(GLuint program, |
| std::vector<int8_t>* result); |
| bool GetTransformFeedbackVaryingHelper( |
| GLuint program, GLuint index, GLsizei bufsize, GLsizei* length, |
| GLint* size, GLenum* type, char* name); |
| bool GetUniformIndicesHelper( |
| GLuint program, GLsizei count, const char* const* names, GLuint* indices); |
| bool GetActiveUniformsivHelper( |
| GLuint program, GLsizei count, const GLuint* indices, |
| GLenum pname, GLint* params); |
| bool GetSyncivHelper( |
| GLsync sync, GLenum pname, GLsizei bufsize, GLsizei* length, |
| GLint* values); |
| bool GetQueryObjectValueHelper( |
| const char* function_name, GLuint id, GLenum pname, GLuint64* params); |
| bool GetProgramInterfaceivHelper( |
| GLuint program, GLenum program_interface, GLenum pname, GLint* params); |
| GLuint GetProgramResourceIndexHelper( |
| GLuint program, GLenum program_interface, const char* name); |
| bool GetProgramResourceNameHelper( |
| GLuint program, GLenum program_interface, GLuint index, GLsizei bufsize, |
| GLsizei* length, char* name); |
| bool GetProgramResourceivHelper( |
| GLuint program, GLenum program_interface, GLuint index, |
| GLsizei prop_count, const GLenum* props, GLsizei bufsize, GLsizei* length, |
| GLint* params); |
| GLint GetProgramResourceLocationHelper( |
| GLuint program, GLenum program_interface, const char* name); |
| |
| const scoped_refptr<ShareGroup>& share_group() const { return share_group_; } |
| |
| GpuControl* gpu_control() { |
| return gpu_control_; |
| } |
| |
| ShareGroupContextData* share_group_context_data() { |
| return &share_group_context_data_; |
| } |
| |
| // QueryTrackerClient implementation. |
| void IssueBeginQuery(GLenum target, |
| GLuint id, |
| uint32_t sync_data_shm_id, |
| uint32_t sync_data_shm_offset) override; |
| void IssueEndQuery(GLenum target, GLuint submit_count) override; |
| void IssueQueryCounter(GLuint id, |
| GLenum target, |
| uint32_t sync_data_shm_id, |
| uint32_t sync_data_shm_offset, |
| GLuint submit_count) override; |
| void IssueSetDisjointValueSync(uint32_t sync_data_shm_id, |
| uint32_t sync_data_shm_offset) override; |
| GLenum GetClientSideGLError() override; |
| CommandBufferHelper* cmd_buffer_helper() override; |
| void SetGLError(GLenum error, |
| const char* function_name, |
| const char* msg) override; |
| |
| CommandBuffer* command_buffer() const; |
| |
| private: |
| friend class GLES2ImplementationTest; |
| friend class VertexArrayObjectManager; |
| friend class QueryTracker; |
| |
| using IdNamespaces = id_namespaces::IdNamespaces; |
| |
| // Used to track whether an extension is available |
| enum ExtensionStatus { |
| kAvailableExtensionStatus, |
| kUnavailableExtensionStatus, |
| kUnknownExtensionStatus |
| }; |
| |
| enum Dimension { |
| k2D, |
| k3D, |
| }; |
| |
| // Base class for mapped resources. |
| struct MappedResource { |
| MappedResource(GLenum _access, int _shm_id, void* mem, unsigned int offset) |
| : access(_access), |
| shm_id(_shm_id), |
| shm_memory(mem), |
| shm_offset(offset) {} |
| |
| // access mode. Currently only GL_WRITE_ONLY is valid |
| GLenum access; |
| |
| // Shared memory ID for buffer. |
| int shm_id; |
| |
| // Address of shared memory |
| void* shm_memory; |
| |
| // Offset of shared memory |
| unsigned int shm_offset; |
| }; |
| |
| // Used to track mapped textures. |
| struct MappedTexture : public MappedResource { |
| MappedTexture( |
| GLenum access, |
| int shm_id, |
| void* shm_mem, |
| unsigned int shm_offset, |
| GLenum _target, |
| GLint _level, |
| GLint _xoffset, |
| GLint _yoffset, |
| GLsizei _width, |
| GLsizei _height, |
| GLenum _format, |
| GLenum _type) |
| : MappedResource(access, shm_id, shm_mem, shm_offset), |
| target(_target), |
| level(_level), |
| xoffset(_xoffset), |
| yoffset(_yoffset), |
| width(_width), |
| height(_height), |
| format(_format), |
| type(_type) { |
| } |
| |
| // These match the arguments to TexSubImage2D. |
| GLenum target; |
| GLint level; |
| GLint xoffset; |
| GLint yoffset; |
| GLsizei width; |
| GLsizei height; |
| GLenum format; |
| GLenum type; |
| }; |
| |
| // Used to track mapped buffers. |
| struct MappedBuffer : public MappedResource { |
| MappedBuffer( |
| GLenum access, |
| int shm_id, |
| void* shm_mem, |
| unsigned int shm_offset, |
| GLenum _target, |
| GLintptr _offset, |
| GLsizeiptr _size) |
| : MappedResource(access, shm_id, shm_mem, shm_offset), |
| target(_target), |
| offset(_offset), |
| size(_size) { |
| } |
| |
| // These match the arguments to BufferSubData. |
| GLenum target; |
| GLintptr offset; |
| GLsizeiptr size; |
| }; |
| |
| struct TextureUnit { |
| TextureUnit() {} |
| |
| // texture currently bound to this unit's GL_TEXTURE_2D with glBindTexture |
| GLuint bound_texture_2d = 0; |
| |
| // texture currently bound to this unit's GL_TEXTURE_CUBE_MAP with |
| // glBindTexture |
| GLuint bound_texture_cube_map = 0; |
| |
| // texture currently bound to this unit's GL_TEXTURE_EXTERNAL_OES with |
| // glBindTexture |
| GLuint bound_texture_external_oes = 0; |
| |
| // texture currently bound to this unit's GL_TEXTURE_RECTANGLE_ARB with |
| // glBindTexture |
| GLuint bound_texture_rectangle_arb = 0; |
| }; |
| |
| // Prevents problematic reentrancy during error callbacks. |
| class DeferErrorCallbacks { |
| public: |
| explicit DeferErrorCallbacks(GLES2Implementation* gles2_implementation); |
| ~DeferErrorCallbacks(); |
| |
| private: |
| GLES2Implementation* gles2_implementation_; |
| }; |
| |
| struct DeferredErrorCallback { |
| // This takes std::string by value and uses std::move in the |
| // implementation, allowing the compiler to achieve zero copies |
| // when passing in a temporary. |
| DeferredErrorCallback(std::string message, int32_t id); |
| |
| std::string message; |
| int32_t id = 0; |
| }; |
| |
| // Checks for single threaded access. |
| class SingleThreadChecker { |
| public: |
| explicit SingleThreadChecker(GLES2Implementation* gles2_implementation); |
| ~SingleThreadChecker(); |
| |
| private: |
| GLES2Implementation* gles2_implementation_; |
| }; |
| |
| // ImplementationBase implementation. |
| void IssueShallowFlush() override; |
| |
| // GpuControlClient implementation. |
| void OnGpuControlLostContext() final; |
| void OnGpuControlLostContextMaybeReentrant() final; |
| void OnGpuControlErrorMessage(const char* message, int32_t id) final; |
| void OnGpuControlSwapBuffersCompleted( |
| const SwapBuffersCompleteParams& params) final; |
| void OnSwapBufferPresented(uint64_t swap_id, |
| const gfx::PresentationFeedback& feedback) final; |
| void OnGpuControlReturnData(base::span<const uint8_t> data) final; |
| |
| void SendErrorMessage(std::string message, int32_t id); |
| void CallDeferredErrorCallbacks(); |
| |
| bool IsChromiumFramebufferMultisampleAvailable(); |
| |
| bool IsExtensionAvailableHelper( |
| const char* extension, ExtensionStatus* status); |
| |
| // Gets the GLError through our wrapper. |
| GLenum GetGLError(); |
| |
| // Sets our wrapper for the GLError. |
| void SetGLErrorInvalidEnum( |
| const char* function_name, GLenum value, const char* label); |
| |
| // Returns the last error and clears it. Useful for debugging. |
| const std::string& GetLastError() { |
| return last_error_; |
| } |
| |
| void AllocateShadowCopiesForReadback(); |
| static void BufferShadowWrittenCallback( |
| const ReadbackBufferShadowTracker::BufferList& buffers, |
| uint64_t serial); |
| |
| // Returns true if id is reserved. |
| bool IsBufferReservedId(GLuint id); |
| bool IsFramebufferReservedId(GLuint id) { return false; } |
| bool IsRenderbufferReservedId(GLuint id) { return false; } |
| bool IsTextureReservedId(GLuint id) { return false; } |
| bool IsVertexArrayReservedId(GLuint id) { return false; } |
| bool IsProgramReservedId(GLuint id) { return false; } |
| bool IsSamplerReservedId(GLuint id) { return false; } |
| bool IsTransformFeedbackReservedId(GLuint id) { return false; } |
| |
| bool UpdateIndexedBufferState(GLenum target, |
| GLuint index, |
| GLuint buffer_id, |
| const char* function_name); |
| void BindBufferHelper(GLenum target, GLuint buffer); |
| void BindBufferBaseHelper(GLenum target, GLuint index, GLuint buffer); |
| void BindBufferRangeHelper(GLenum target, GLuint index, GLuint buffer, |
| GLintptr offset, GLsizeiptr size); |
| void BindFramebufferHelper(GLenum target, GLuint framebuffer); |
| void BindRenderbufferHelper(GLenum target, GLuint renderbuffer); |
| void BindSamplerHelper(GLuint unit, GLuint sampler); |
| void BindTextureHelper(GLenum target, GLuint texture); |
| void BindTransformFeedbackHelper(GLenum target, GLuint transformfeedback); |
| void BindVertexArrayOESHelper(GLuint array); |
| void UseProgramHelper(GLuint program); |
| |
| void BindBufferStub(GLenum target, GLuint buffer); |
| void BindBufferBaseStub(GLenum target, GLuint index, GLuint buffer); |
| void BindBufferRangeStub(GLenum target, GLuint index, GLuint buffer, |
| GLintptr offset, GLsizeiptr size); |
| void BindRenderbufferStub(GLenum target, GLuint renderbuffer); |
| void BindTextureStub(GLenum target, GLuint texture); |
| |
| void GenBuffersHelper(GLsizei n, const GLuint* buffers); |
| void GenFramebuffersHelper(GLsizei n, const GLuint* framebuffers); |
| void GenRenderbuffersHelper(GLsizei n, const GLuint* renderbuffers); |
| void GenTexturesHelper(GLsizei n, const GLuint* textures); |
| void GenVertexArraysOESHelper(GLsizei n, const GLuint* arrays); |
| void GenQueriesEXTHelper(GLsizei n, const GLuint* queries); |
| void GenSamplersHelper(GLsizei n, const GLuint* samplers); |
| void GenTransformFeedbacksHelper(GLsizei n, const GLuint* transformfeedbacks); |
| |
| void DeleteBuffersHelper(GLsizei n, const GLuint* buffers); |
| void DeleteFramebuffersHelper(GLsizei n, const GLuint* framebuffers); |
| void DeleteRenderbuffersHelper(GLsizei n, const GLuint* renderbuffers); |
| void DeleteTexturesHelper(GLsizei n, const GLuint* textures); |
| void UnbindTexturesHelper(GLsizei n, const GLuint* textures); |
| bool DeleteProgramHelper(GLuint program); |
| bool DeleteShaderHelper(GLuint shader); |
| void DeleteQueriesEXTHelper(GLsizei n, const GLuint* queries); |
| void DeleteVertexArraysOESHelper(GLsizei n, const GLuint* arrays); |
| void DeleteSamplersHelper(GLsizei n, const GLuint* samplers); |
| void DeleteTransformFeedbacksHelper( |
| GLsizei n, const GLuint* transformfeedbacks); |
| void DeleteSyncHelper(GLsync sync); |
| |
| void DeleteBuffersStub(GLsizei n, const GLuint* buffers); |
| void DeleteRenderbuffersStub(GLsizei n, const GLuint* renderbuffers); |
| void DeleteTexturesStub(GLsizei n, const GLuint* textures); |
| void DeletePathsCHROMIUMStub(GLuint first_client_id, GLsizei range); |
| void DeleteProgramStub(GLsizei n, const GLuint* programs); |
| void DeleteShaderStub(GLsizei n, const GLuint* shaders); |
| void DeleteSamplersStub(GLsizei n, const GLuint* samplers); |
| void DeleteSyncStub(GLsizei n, const GLuint* syncs); |
| void DestroyGpuFenceCHROMIUMHelper(GLuint client_id); |
| |
| void BufferDataHelper( |
| GLenum target, GLsizeiptr size, const void* data, GLenum usage); |
| void BufferSubDataHelper( |
| GLenum target, GLintptr offset, GLsizeiptr size, const void* data); |
| void BufferSubDataHelperImpl( |
| GLenum target, GLintptr offset, GLsizeiptr size, const void* data, |
| ScopedTransferBufferPtr* buffer); |
| |
| void MultiDrawArraysWEBGLHelper(GLenum mode, |
| const GLint* firsts, |
| const GLsizei* counts, |
| GLsizei drawcount); |
| |
| void MultiDrawArraysInstancedWEBGLHelper(GLenum mode, |
| const GLint* firsts, |
| const GLsizei* counts, |
| const GLsizei* instanceCounts, |
| GLsizei drawcount); |
| |
| void MultiDrawElementsWEBGLHelper(GLenum mode, |
| const GLsizei* counts, |
| GLenum type, |
| const GLsizei* offsets, |
| GLsizei drawcount); |
| |
| void MultiDrawElementsInstancedWEBGLHelper(GLenum mode, |
| const GLsizei* counts, |
| GLenum type, |
| const GLsizei* offsets, |
| const GLsizei* instanceCounts, |
| GLsizei drawcount); |
| |
| GLuint CreateImageCHROMIUMHelper(ClientBuffer buffer, |
| GLsizei width, |
| GLsizei height, |
| GLenum internalformat); |
| void DestroyImageCHROMIUMHelper(GLuint image_id); |
| |
| // Helper for GetVertexAttrib |
| bool GetVertexAttribHelper(GLuint index, GLenum pname, uint32_t* param); |
| |
| GLuint GetMaxValueInBufferCHROMIUMHelper( |
| GLuint buffer_id, GLsizei count, GLenum type, GLuint offset); |
| |
| void WaitAllAsyncTexImage2DCHROMIUMHelper(); |
| |
| void RestoreElementAndArrayBuffers(bool restore); |
| void RestoreArrayBuffer(bool restrore); |
| |
| // The pixels pointer should already account for unpack skip |
| // images/rows/pixels. |
| void TexSubImage2DImpl(GLenum target, |
| GLint level, |
| GLint xoffset, |
| GLint yoffset, |
| GLsizei width, |
| GLsizei height, |
| GLenum format, |
| GLenum type, |
| uint32_t unpadded_row_size, |
| const void* pixels, |
| uint32_t pixels_padded_row_size, |
| GLboolean internal, |
| ScopedTransferBufferPtr* buffer, |
| uint32_t buffer_padded_row_size); |
| void TexSubImage3DImpl(GLenum target, |
| GLint level, |
| GLint xoffset, |
| GLint yoffset, |
| GLint zoffset, |
| GLsizei width, |
| GLsizei height, |
| GLsizei depth, |
| GLenum format, |
| GLenum type, |
| uint32_t unpadded_row_size, |
| const void* pixels, |
| uint32_t pixels_padded_row_size, |
| GLboolean internal, |
| ScopedTransferBufferPtr* buffer, |
| uint32_t buffer_padded_row_size); |
| |
| // Helpers for query functions. |
| bool GetHelper(GLenum pname, GLint* params); |
| GLuint GetBoundBufferHelper(GLenum target); |
| bool GetBooleanvHelper(GLenum pname, GLboolean* params); |
| bool GetBufferParameteri64vHelper( |
| GLenum target, GLenum pname, GLint64* params); |
| bool GetBufferParameterivHelper(GLenum target, GLenum pname, GLint* params); |
| bool GetFloatvHelper(GLenum pname, GLfloat* params); |
| bool GetFramebufferAttachmentParameterivHelper( |
| GLenum target, GLenum attachment, GLenum pname, GLint* params); |
| bool GetInteger64vHelper(GLenum pname, GLint64* params); |
| bool GetIntegervHelper(GLenum pname, GLint* params); |
| bool GetIntegeri_vHelper(GLenum pname, GLuint index, GLint* data); |
| bool GetInteger64i_vHelper(GLenum pname, GLuint index, GLint64* data); |
| bool GetInternalformativHelper( |
| GLenum target, GLenum format, GLenum pname, GLsizei bufSize, |
| GLint* params); |
| bool GetProgramivHelper(GLuint program, GLenum pname, GLint* params); |
| bool GetSamplerParameterfvHelper( |
| GLuint sampler, GLenum pname, GLfloat* params); |
| bool GetSamplerParameterivHelper( |
| GLuint sampler, GLenum pname, GLint* params); |
| bool GetRenderbufferParameterivHelper( |
| GLenum target, GLenum pname, GLint* params); |
| bool GetShaderivHelper(GLuint shader, GLenum pname, GLint* params); |
| bool GetTexParameterfvHelper(GLenum target, GLenum pname, GLfloat* params); |
| bool GetTexParameterivHelper(GLenum target, GLenum pname, GLint* params); |
| const GLubyte* GetStringHelper(GLenum name); |
| |
| bool IsExtensionAvailable(const char* ext); |
| |
| // Caches certain capabilties state. Return true if cached. |
| bool SetCapabilityState(GLenum cap, bool enabled); |
| |
| IdHandlerInterface* GetIdHandler(SharedIdNamespaces id_namespace) const; |
| RangeIdHandlerInterface* GetRangeIdHandler(int id_namespace) const; |
| // IdAllocators for objects that can't be shared among contexts. |
| IdAllocator* GetIdAllocator(IdNamespaces id_namespace) const; |
| |
| void FinishHelper(); |
| void FlushHelper(); |
| |
| void RunIfContextNotLost(base::OnceClosure callback); |
| |
| // Validate if an offset is valid, i.e., non-negative and fit into 32-bit. |
| // If not, generate an approriate error, and return false. |
| bool ValidateOffset(const char* func, GLintptr offset); |
| |
| // Validate if a size is valid, i.e., non-negative and fit into 32-bit. |
| // If not, generate an approriate error, and return false. |
| bool ValidateSize(const char* func, GLsizeiptr offset); |
| |
| // Remove the transfer buffer from the buffer tracker. For buffers used |
| // asynchronously the memory is free:ed if the upload has completed. For |
| // other buffers, the memory is either free:ed immediately or free:ed pending |
| // a token. |
| void RemoveTransferBuffer(BufferTracker::Buffer* buffer); |
| |
| bool GetBoundPixelTransferBuffer( |
| GLenum target, const char* function_name, GLuint* buffer_id); |
| BufferTracker::Buffer* GetBoundPixelTransferBufferIfValid( |
| GLuint buffer_id, |
| const char* function_name, GLuint offset, GLsizei size); |
| |
| void OnBufferWrite(GLenum target); |
| |
| // Pack 2D arrays of char into a bucket. |
| // Helper function for ShaderSource(), TransformFeedbackVaryings(), etc. |
| bool PackStringsToBucket(GLsizei count, |
| const char* const* str, |
| const GLint* length, |
| const char* func_name); |
| |
| const std::string& GetLogPrefix() const; |
| |
| bool PrepareInstancedPathCommand(const char* function_name, |
| GLsizei num_paths, |
| GLenum path_name_type, |
| const void* paths, |
| GLenum transform_type, |
| const GLfloat* transform_values, |
| ScopedTransferBufferPtr* buffer, |
| uint32_t* out_paths_shm_id, |
| uint32_t* out_paths_offset, |
| uint32_t* out_transforms_shm_id, |
| uint32_t* out_transforms_offset); |
| |
| // Set to 1 to have the client fail when a GL error is generated. |
| // This helps find bugs in the renderer since the debugger stops on the error. |
| #if DCHECK_IS_ON() |
| #if 0 |
| #define GL_CLIENT_FAIL_GL_ERRORS |
| #endif |
| #endif |
| |
| #if defined(GL_CLIENT_FAIL_GL_ERRORS) |
| void CheckGLError(); |
| void FailGLError(GLenum error); |
| #else |
| void CheckGLError() { } |
| void FailGLError(GLenum /* error */) { } |
| #endif |
| |
| void RemoveMappedBufferRangeByTarget(GLenum target); |
| void RemoveMappedBufferRangeById(GLuint buffer); |
| void ClearMappedBufferRangeMap(); |
| |
| void DrawElementsImpl(GLenum mode, GLsizei count, GLenum type, |
| const void* indices, const char* func_name); |
| void UpdateCachedExtensionsIfNeeded(); |
| void InvalidateCachedExtensions(); |
| |
| PixelStoreParams GetUnpackParameters(Dimension dimension); |
| |
| uint64_t PrepareNextSwapId(SwapCompletedCallback complete_callback, |
| PresentationCallback present_callback); |
| |
| GLES2Util util_; |
| GLES2CmdHelper* helper_; |
| std::string last_error_; |
| DebugMarkerManager debug_marker_manager_; |
| std::string this_in_hex_; |
| |
| base::queue<int32_t> swap_buffers_tokens_; |
| |
| ExtensionStatus chromium_framebuffer_multisample_; |
| |
| GLStaticState static_state_; |
| ClientContextState state_; |
| |
| // pack alignment as last set by glPixelStorei |
| GLint pack_alignment_; |
| |
| // pack row length as last set by glPixelStorei |
| GLint pack_row_length_; |
| |
| // pack skip pixels as last set by glPixelStorei |
| GLint pack_skip_pixels_; |
| |
| // pack skip rows as last set by glPixelStorei |
| GLint pack_skip_rows_; |
| |
| // unpack alignment as last set by glPixelStorei |
| GLint unpack_alignment_; |
| |
| // unpack row length as last set by glPixelStorei |
| GLint unpack_row_length_; |
| |
| // unpack image height as last set by glPixelStorei |
| GLint unpack_image_height_; |
| |
| // unpack skip rows as last set by glPixelStorei |
| GLint unpack_skip_rows_; |
| |
| // unpack skip pixels as last set by glPixelStorei |
| GLint unpack_skip_pixels_; |
| |
| // unpack skip images as last set by glPixelStorei |
| GLint unpack_skip_images_; |
| |
| std::unique_ptr<TextureUnit[]> texture_units_; |
| |
| // 0 to gl_state_.max_combined_texture_image_units. |
| GLuint active_texture_unit_; |
| |
| GLuint bound_framebuffer_; |
| GLuint bound_read_framebuffer_; |
| GLuint bound_renderbuffer_; |
| |
| // The program in use by glUseProgram |
| GLuint current_program_; |
| |
| GLuint bound_array_buffer_; |
| GLuint bound_atomic_counter_buffer_; |
| GLuint bound_copy_read_buffer_; |
| GLuint bound_copy_write_buffer_; |
| GLuint bound_pixel_pack_buffer_; |
| GLuint bound_pixel_unpack_buffer_; |
| GLuint bound_shader_storage_buffer_; |
| GLuint bound_transform_feedback_buffer_; |
| GLuint bound_uniform_buffer_; |
| // We don't cache the currently bound transform feedback buffer, because |
| // it is part of the current transform feedback object. Caching the transform |
| // feedback object state correctly requires predicting if a call to |
| // glBeginTransformFeedback will succeed or fail, which in turn requires |
| // caching a whole bunch of other states such as the transform feedback |
| // varyings of the current program. |
| |
| // The currently bound pixel transfer buffers. |
| GLuint bound_pixel_pack_transfer_buffer_id_; |
| GLuint bound_pixel_unpack_transfer_buffer_id_; |
| |
| // Client side management for vertex array objects. Needed to correctly |
| // track client side arrays. |
| std::unique_ptr<VertexArrayObjectManager> vertex_array_object_manager_; |
| |
| GLuint reserved_ids_[2]; |
| |
| // Current GL error bits. |
| uint32_t error_bits_; |
| |
| LogSettings log_settings_; |
| |
| // When true, the context is lost when a GL_OUT_OF_MEMORY error occurs. |
| const bool lose_context_when_out_of_memory_; |
| |
| // Whether or not to support client side arrays. |
| const bool support_client_side_arrays_; |
| |
| // Used to check for single threaded access. |
| int use_count_; |
| |
| // Changed every time a flush or finish occurs. |
| uint32_t flush_id_; |
| |
| // Avoid recycling of client-allocated GpuFence IDs by saving the |
| // last-allocated one and requesting the next one to be higher than that. |
| // This will wrap around as needed, but the space should be plenty big enough |
| // to avoid collisions. |
| uint32_t last_gpu_fence_id_ = 0; |
| |
| // Maximum amount of extra memory from the mapped memory pool to use when |
| // needing to transfer something exceeding the default transfer buffer. |
| uint32_t max_extra_transfer_buffer_size_; |
| |
| // Set of strings returned from glGetString. We need to cache these because |
| // the pointer passed back to the client has to remain valid for eternity. |
| std::set<std::string> gl_strings_; |
| |
| typedef std::map<const void*, MappedBuffer> MappedBufferMap; |
| MappedBufferMap mapped_buffers_; |
| |
| // TODO(zmo): Consolidate |mapped_buffers_| and |mapped_buffer_range_map_|. |
| typedef std::unordered_map<GLuint, MappedBuffer> MappedBufferRangeMap; |
| MappedBufferRangeMap mapped_buffer_range_map_; |
| |
| typedef std::map<const void*, MappedTexture> MappedTextureMap; |
| MappedTextureMap mapped_textures_; |
| |
| scoped_refptr<ShareGroup> share_group_; |
| ShareGroupContextData share_group_context_data_; |
| |
| std::unique_ptr<IdAllocator> |
| id_allocators_[static_cast<int>(IdNamespaces::kNumIdNamespaces)]; |
| |
| std::unique_ptr<BufferTracker> buffer_tracker_; |
| std::unique_ptr<ReadbackBufferShadowTracker> readback_buffer_shadow_tracker_; |
| |
| base::Optional<ScopedMappedMemoryPtr> font_mapped_buffer_; |
| base::Optional<ScopedTransferBufferPtr> raster_mapped_buffer_; |
| |
| base::RepeatingCallback<void(const char*, int32_t)> error_message_callback_; |
| bool deferring_error_callbacks_ = false; |
| std::deque<DeferredErrorCallback> deferred_error_callbacks_; |
| |
| int current_trace_stack_; |
| |
| // Flag to indicate whether the implementation can retain resources, or |
| // whether it should aggressively free them. |
| bool aggressively_free_resources_; |
| |
| // Result of last GetString(GL_EXTENSIONS), used to keep |
| // GetString(GL_EXTENSIONS), GetStringi(GL_EXTENSIONS, index) and |
| // GetIntegerv(GL_NUM_EXTENSIONS) in sync. This points to gl_strings, valid |
| // forever. |
| const char* cached_extension_string_; |
| |
| // Populated if cached_extension_string_ != nullptr. These point to |
| // gl_strings, valid forever. |
| std::vector<const char*> cached_extensions_; |
| |
| // The next swap ID to send. |
| uint64_t swap_id_ = 0; |
| // A map of swap IDs to callbacks to run when that ID completes. |
| base::flat_map<uint64_t, SwapCompletedCallback> pending_swap_callbacks_; |
| base::flat_map<uint64_t, PresentationCallback> |
| pending_presentation_callbacks_; |
| |
| std::string last_active_url_; |
| |
| base::WeakPtrFactory<GLES2Implementation> weak_ptr_factory_; |
| |
| DISALLOW_COPY_AND_ASSIGN(GLES2Implementation); |
| }; |
| |
| inline bool GLES2Implementation::GetBufferParameteri64vHelper( |
| GLenum /* target */, GLenum /* pname */, GLint64* /* params */) { |
| return false; |
| } |
| |
| inline bool GLES2Implementation::GetBufferParameterivHelper( |
| GLenum /* target */, GLenum /* pname */, GLint* /* params */) { |
| return false; |
| } |
| |
| inline bool GLES2Implementation::GetFramebufferAttachmentParameterivHelper( |
| GLenum /* target */, |
| GLenum /* attachment */, |
| GLenum /* pname */, |
| GLint* /* params */) { |
| return false; |
| } |
| |
| inline bool GLES2Implementation::GetRenderbufferParameterivHelper( |
| GLenum /* target */, GLenum /* pname */, GLint* /* params */) { |
| return false; |
| } |
| |
| inline bool GLES2Implementation::GetShaderivHelper( |
| GLuint /* shader */, GLenum /* pname */, GLint* /* params */) { |
| return false; |
| } |
| |
| inline bool GLES2Implementation::GetTexParameterfvHelper( |
| GLenum /* target */, GLenum /* pname */, GLfloat* /* params */) { |
| return false; |
| } |
| |
| inline bool GLES2Implementation::GetTexParameterivHelper( |
| GLenum /* target */, GLenum /* pname */, GLint* /* params */) { |
| return false; |
| } |
| |
| } // namespace gles2 |
| } // namespace gpu |
| |
| #endif // GPU_COMMAND_BUFFER_CLIENT_GLES2_IMPLEMENTATION_H_ |