// 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. // This file defines the GLES2 command buffer commands. #ifndef GPU_COMMAND_BUFFER_COMMON_GLES2_CMD_FORMAT_H_ #define GPU_COMMAND_BUFFER_COMMON_GLES2_CMD_FORMAT_H_ #include #include #include "base/safe_numerics.h" #include "gpu/command_buffer/common/bitfield_helpers.h" #include "gpu/command_buffer/common/cmd_buffer_common.h" #include "gpu/command_buffer/common/gles2_cmd_ids.h" #include "gpu/command_buffer/common/types.h" // GL types are forward declared to avoid including the GL headers. The problem // is determining which GL headers to include from code that is common to the // client and service sides (GLES2 or one of several GL implementations). typedef unsigned int GLenum; typedef unsigned int GLbitfield; typedef unsigned int GLuint; typedef int GLint; typedef int GLsizei; typedef unsigned char GLboolean; typedef signed char GLbyte; typedef short GLshort; typedef unsigned char GLubyte; typedef unsigned short GLushort; typedef unsigned long GLulong; typedef float GLfloat; typedef float GLclampf; typedef double GLdouble; typedef double GLclampd; typedef void GLvoid; typedef khronos_intptr_t GLintptr; typedef khronos_ssize_t GLsizeiptr; namespace gpu { namespace gles2 { #pragma pack(push, 1) namespace id_namespaces { // These are used when contexts share resources. enum IdNamespaces { kBuffers, kFramebuffers, kProgramsAndShaders, kRenderbuffers, kTextures, kQueries, kVertexArrays, kNumIdNamespaces }; // These numbers must not change COMPILE_ASSERT(kBuffers == 0, kBuffers_is_not_0); COMPILE_ASSERT(kFramebuffers == 1, kFramebuffers_is_not_1); COMPILE_ASSERT(kProgramsAndShaders == 2, kProgramsAndShaders_is_not_2); COMPILE_ASSERT(kRenderbuffers == 3, kRenderbuffers_is_not_3); COMPILE_ASSERT(kTextures == 4, kTextures_is_not_4); } // namespace id_namespaces // Used for some glGetXXX commands that return a result through a pointer. We // need to know if the command succeeded or not and the size of the result. If // the command failed its result size will 0. template struct SizedResult { typedef T Type; T* GetData() { return static_cast(static_cast(&data)); } // Returns the total size in bytes of the SizedResult for a given number of // results including the size field. static size_t ComputeSize(size_t num_results) { return sizeof(T) * num_results + sizeof(uint32); // NOLINT } // Returns the total size in bytes of the SizedResult for a given size of // results. static size_t ComputeSizeFromBytes(size_t size_of_result_in_bytes) { return size_of_result_in_bytes + sizeof(uint32); // NOLINT } // Returns the maximum number of results for a given buffer size. static uint32 ComputeMaxResults(size_t size_of_buffer) { return (size_of_buffer >= sizeof(uint32)) ? ((size_of_buffer - sizeof(uint32)) / sizeof(T)) : 0; // NOLINT } // Set the size for a given number of results. void SetNumResults(size_t num_results) { size = sizeof(T) * num_results; // NOLINT } // Get the number of elements in the result int32 GetNumResults() const { return size / sizeof(T); // NOLINT } // Copy the result. void CopyResult(void* dst) const { memcpy(dst, &data, size); } uint32 size; // in bytes. int32 data; // this is just here to get an offset. }; COMPILE_ASSERT(sizeof(SizedResult) == 8, SizedResult_size_not_8); COMPILE_ASSERT(offsetof(SizedResult, size) == 0, OffsetOf_SizedResult_size_not_0); COMPILE_ASSERT(offsetof(SizedResult, data) == 4, OffsetOf_SizedResult_data_not_4); // The data for one attrib or uniform from GetProgramInfoCHROMIUM. struct ProgramInput { uint32 type; // The type (GL_VEC3, GL_MAT3, GL_SAMPLER_2D, etc. int32 size; // The size (how big the array is for uniforms) uint32 location_offset; // offset from ProgramInfoHeader to 'size' locations // for uniforms, 1 for attribs. uint32 name_offset; // offset from ProgrmaInfoHeader to start of name. uint32 name_length; // length of the name. }; // The format of the bucket filled out by GetProgramInfoCHROMIUM struct ProgramInfoHeader { uint32 link_status; uint32 num_attribs; uint32 num_uniforms; // ProgramInput inputs[num_attribs + num_uniforms]; }; // The format of QuerySync used by EXT_occlusion_query_boolean struct QuerySync { void Reset() { process_count = 0; result = 0; } uint32 process_count; uint64 result; }; COMPILE_ASSERT(sizeof(ProgramInput) == 20, ProgramInput_size_not_20); COMPILE_ASSERT(offsetof(ProgramInput, type) == 0, OffsetOf_ProgramInput_type_not_0); COMPILE_ASSERT(offsetof(ProgramInput, size) == 4, OffsetOf_ProgramInput_size_not_4); COMPILE_ASSERT(offsetof(ProgramInput, location_offset) == 8, OffsetOf_ProgramInput_location_offset_not_8); COMPILE_ASSERT(offsetof(ProgramInput, name_offset) == 12, OffsetOf_ProgramInput_name_offset_not_12); COMPILE_ASSERT(offsetof(ProgramInput, name_length) == 16, OffsetOf_ProgramInput_name_length_not_16); COMPILE_ASSERT(sizeof(ProgramInfoHeader) == 12, ProgramInfoHeader_size_not_12); COMPILE_ASSERT(offsetof(ProgramInfoHeader, link_status) == 0, OffsetOf_ProgramInfoHeader_link_status_not_0); COMPILE_ASSERT(offsetof(ProgramInfoHeader, num_attribs) == 4, OffsetOf_ProgramInfoHeader_num_attribs_not_4); COMPILE_ASSERT(offsetof(ProgramInfoHeader, num_uniforms) == 8, OffsetOf_ProgramInfoHeader_num_uniforms_not_8); namespace cmds { #include "../common/gles2_cmd_format_autogen.h" // These are hand written commands. // TODO(gman): Attempt to make these auto-generated. struct GetAttribLocation { typedef GetAttribLocation ValueType; static const CommandId kCmdId = kGetAttribLocation; static const cmd::ArgFlags kArgFlags = cmd::kFixed; typedef GLint Result; static uint32 ComputeSize() { return static_cast(sizeof(ValueType)); // NOLINT } void SetHeader() { header.SetCmd(); } void Init( GLuint _program, uint32 _name_shm_id, uint32 _name_shm_offset, uint32 _location_shm_id, uint32 _location_shm_offset, uint32 _data_size) { SetHeader(); program = _program; name_shm_id = _name_shm_id; name_shm_offset = _name_shm_offset; location_shm_id = _location_shm_id; location_shm_offset = _location_shm_offset; data_size = _data_size; } void* Set( void* cmd, GLuint _program, uint32 _name_shm_id, uint32 _name_shm_offset, uint32 _location_shm_id, uint32 _location_shm_offset, uint32 _data_size) { static_cast( cmd)->Init( _program, _name_shm_id, _name_shm_offset, _location_shm_id, _location_shm_offset, _data_size); return NextCmdAddress(cmd); } CommandHeader header; uint32 program; uint32 name_shm_id; uint32 name_shm_offset; uint32 location_shm_id; uint32 location_shm_offset; uint32 data_size; }; COMPILE_ASSERT(sizeof(GetAttribLocation) == 28, Sizeof_GetAttribLocation_is_not_28); COMPILE_ASSERT(offsetof(GetAttribLocation, header) == 0, OffsetOf_GetAttribLocation_header_not_0); COMPILE_ASSERT(offsetof(GetAttribLocation, program) == 4, OffsetOf_GetAttribLocation_program_not_4); COMPILE_ASSERT(offsetof(GetAttribLocation, name_shm_id) == 8, OffsetOf_GetAttribLocation_name_shm_id_not_8); COMPILE_ASSERT(offsetof(GetAttribLocation, name_shm_offset) == 12, OffsetOf_GetAttribLocation_name_shm_offset_not_12); COMPILE_ASSERT(offsetof(GetAttribLocation, location_shm_id) == 16, OffsetOf_GetAttribLocation_location_shm_id_not_16); COMPILE_ASSERT(offsetof(GetAttribLocation, location_shm_offset) == 20, OffsetOf_GetAttribLocation_location_shm_offset_not_20); COMPILE_ASSERT(offsetof(GetAttribLocation, data_size) == 24, OffsetOf_GetAttribLocation_data_size_not_24); struct GetAttribLocationImmediate { typedef GetAttribLocationImmediate ValueType; static const CommandId kCmdId = kGetAttribLocationImmediate; static const cmd::ArgFlags kArgFlags = cmd::kAtLeastN; typedef GLint Result; static uint32 ComputeDataSize(const char* s) { return base::checked_numeric_cast(strlen(s)); } static uint32 ComputeSize(const char* s) { return base::checked_numeric_cast(sizeof(ValueType) + ComputeDataSize(s)); } void SetHeader(const char* s) { header.SetCmdByTotalSize(ComputeSize(s)); } void Init( GLuint _program, const char* _name, uint32 _location_shm_id, uint32 _location_shm_offset) { SetHeader(_name); program = _program; location_shm_id = _location_shm_id; location_shm_offset = _location_shm_offset; data_size = ComputeDataSize(_name); memcpy(ImmediateDataAddress(this), _name, data_size); } void* Set( void* cmd, GLuint _program, const char* _name, uint32 _location_shm_id, uint32 _location_shm_offset) { uint32 total_size = ComputeSize(_name); static_cast( cmd)->Init(_program, _name, _location_shm_id, _location_shm_offset); return NextImmediateCmdAddressTotalSize(cmd, total_size); } CommandHeader header; uint32 program; uint32 location_shm_id; uint32 location_shm_offset; uint32 data_size; }; COMPILE_ASSERT(sizeof(GetAttribLocationImmediate) == 20, Sizeof_GetAttribLocationImmediate_is_not_20); COMPILE_ASSERT(offsetof(GetAttribLocationImmediate, header) == 0, OffsetOf_GetAttribLocationImmediate_header_not_0); COMPILE_ASSERT(offsetof(GetAttribLocationImmediate, program) == 4, OffsetOf_GetAttribLocationImmediate_program_not_4); COMPILE_ASSERT(offsetof(GetAttribLocationImmediate, location_shm_id) == 8, OffsetOf_GetAttribLocationImmediate_location_shm_id_not_8); COMPILE_ASSERT(offsetof(GetAttribLocationImmediate, location_shm_offset) == 12, OffsetOf_GetAttribLocationImmediate_location_shm_offset_not_12); COMPILE_ASSERT(offsetof(GetAttribLocationImmediate, data_size) == 16, OffsetOf_GetAttribLocationImmediate_data_size_not_16); struct GetAttribLocationBucket { typedef GetAttribLocationBucket ValueType; static const CommandId kCmdId = kGetAttribLocationBucket; static const cmd::ArgFlags kArgFlags = cmd::kFixed; typedef GLint Result; static uint32 ComputeSize() { return static_cast(sizeof(ValueType)); // NOLINT } void SetHeader() { header.SetCmd(); } void Init( GLuint _program, uint32 _name_bucket_id, uint32 _location_shm_id, uint32 _location_shm_offset) { SetHeader(); program = _program; name_bucket_id = _name_bucket_id; location_shm_id = _location_shm_id; location_shm_offset = _location_shm_offset; } void* Set( void* cmd, GLuint _program, uint32 _name_bucket_id, uint32 _location_shm_id, uint32 _location_shm_offset) { static_cast( cmd)->Init( _program, _name_bucket_id, _location_shm_id, _location_shm_offset); return NextCmdAddress(cmd); } CommandHeader header; uint32 program; uint32 name_bucket_id; uint32 location_shm_id; uint32 location_shm_offset; }; COMPILE_ASSERT(sizeof(GetAttribLocationBucket) == 20, Sizeof_GetAttribLocationBucket_is_not_24); COMPILE_ASSERT(offsetof(GetAttribLocationBucket, header) == 0, OffsetOf_GetAttribLocationBucket_header_not_0); COMPILE_ASSERT(offsetof(GetAttribLocationBucket, program) == 4, OffsetOf_GetAttribLocationBucket_program_not_4); COMPILE_ASSERT(offsetof(GetAttribLocationBucket, name_bucket_id) == 8, OffsetOf_GetAttribLocationBucket_name_bucket_id_not_8); COMPILE_ASSERT(offsetof(GetAttribLocationBucket, location_shm_id) == 12, OffsetOf_GetAttribLocationBucket_location_shm_id_not_12); COMPILE_ASSERT(offsetof(GetAttribLocationBucket, location_shm_offset) == 16, OffsetOf_GetAttribLocationBucket_location_shm_offset_not_16); struct GetUniformLocation { typedef GetUniformLocation ValueType; static const CommandId kCmdId = kGetUniformLocation; static const cmd::ArgFlags kArgFlags = cmd::kFixed; typedef GLint Result; static uint32 ComputeSize() { return static_cast(sizeof(ValueType)); // NOLINT } void SetHeader() { header.SetCmd(); } void Init( GLuint _program, uint32 _name_shm_id, uint32 _name_shm_offset, uint32 _location_shm_id, uint32 _location_shm_offset, uint32 _data_size) { SetHeader(); program = _program; name_shm_id = _name_shm_id; name_shm_offset = _name_shm_offset; location_shm_id = _location_shm_id; location_shm_offset = _location_shm_offset; data_size = _data_size; } void* Set( void* cmd, GLuint _program, uint32 _name_shm_id, uint32 _name_shm_offset, uint32 _location_shm_id, uint32 _location_shm_offset, uint32 _data_size) { static_cast( cmd)->Init( _program, _name_shm_id, _name_shm_offset, _location_shm_id, _location_shm_offset, _data_size); return NextCmdAddress(cmd); } CommandHeader header; uint32 program; uint32 name_shm_id; uint32 name_shm_offset; uint32 location_shm_id; uint32 location_shm_offset; uint32 data_size; }; COMPILE_ASSERT(sizeof(GetUniformLocation) == 28, Sizeof_GetUniformLocation_is_not_28); COMPILE_ASSERT(offsetof(GetUniformLocation, header) == 0, OffsetOf_GetUniformLocation_header_not_0); COMPILE_ASSERT(offsetof(GetUniformLocation, program) == 4, OffsetOf_GetUniformLocation_program_not_4); COMPILE_ASSERT(offsetof(GetUniformLocation, name_shm_id) == 8, OffsetOf_GetUniformLocation_name_shm_id_not_8); COMPILE_ASSERT(offsetof(GetUniformLocation, name_shm_offset) == 12, OffsetOf_GetUniformLocation_name_shm_offset_not_12); COMPILE_ASSERT(offsetof(GetUniformLocation, location_shm_id) == 16, OffsetOf_GetUniformLocation_location_shm_id_not_16); COMPILE_ASSERT(offsetof(GetUniformLocation, location_shm_offset) == 20, OffsetOf_GetUniformLocation_location_shm_offset_not_20); COMPILE_ASSERT(offsetof(GetUniformLocation, data_size) == 24, OffsetOf_GetUniformLocation_data_size_not_24); struct GetUniformLocationImmediate { typedef GetUniformLocationImmediate ValueType; static const CommandId kCmdId = kGetUniformLocationImmediate; static const cmd::ArgFlags kArgFlags = cmd::kAtLeastN; typedef GLint Result; static uint32 ComputeDataSize(const char* s) { return base::checked_numeric_cast(strlen(s)); } static uint32 ComputeSize(const char* s) { return base::checked_numeric_cast(sizeof(ValueType) + ComputeDataSize(s)); } void SetHeader(const char* s) { header.SetCmdByTotalSize(ComputeSize(s)); } void Init( GLuint _program, const char* _name, uint32 _location_shm_id, uint32 _location_shm_offset) { SetHeader(_name); program = _program; location_shm_id = _location_shm_id; location_shm_offset = _location_shm_offset; data_size = ComputeDataSize(_name); memcpy(ImmediateDataAddress(this), _name, data_size); } void* Set( void* cmd, GLuint _program, const char* _name, uint32 _location_shm_id, uint32 _location_shm_offset) { uint32 total_size = ComputeSize(_name); static_cast( cmd)->Init(_program, _name, _location_shm_id, _location_shm_offset); return NextImmediateCmdAddressTotalSize(cmd, total_size); } CommandHeader header; uint32 program; uint32 location_shm_id; uint32 location_shm_offset; uint32 data_size; }; COMPILE_ASSERT(sizeof(GetUniformLocationImmediate) == 20, Sizeof_GetUniformLocationImmediate_is_not_20); COMPILE_ASSERT(offsetof(GetUniformLocationImmediate, header) == 0, OffsetOf_GetUniformLocationImmediate_header_not_0); COMPILE_ASSERT(offsetof(GetUniformLocationImmediate, program) == 4, OffsetOf_GetUniformLocationImmediate_program_not_4); COMPILE_ASSERT(offsetof(GetUniformLocationImmediate, location_shm_id) == 8, OffsetOf_GetUniformLocationImmediate_location_shm_id_not_8); COMPILE_ASSERT( offsetof(GetUniformLocationImmediate, location_shm_offset) == 12, OffsetOf_GetUniformLocationImmediate_location_shm_offset_not_12); COMPILE_ASSERT(offsetof(GetUniformLocationImmediate, data_size) == 16, OffsetOf_GetUniformLocationImmediate_data_size_not_16); struct GetUniformLocationBucket { typedef GetUniformLocationBucket ValueType; static const CommandId kCmdId = kGetUniformLocationBucket; static const cmd::ArgFlags kArgFlags = cmd::kFixed; typedef GLint Result; static uint32 ComputeSize() { return static_cast(sizeof(ValueType)); // NOLINT } void SetHeader() { header.SetCmd(); } void Init( GLuint _program, uint32 _name_bucket_id, uint32 _location_shm_id, uint32 _location_shm_offset) { SetHeader(); program = _program; name_bucket_id = _name_bucket_id; location_shm_id = _location_shm_id; location_shm_offset = _location_shm_offset; } void* Set( void* cmd, GLuint _program, uint32 _name_bucket_id, uint32 _location_shm_id, uint32 _location_shm_offset) { static_cast( cmd)->Init( _program, _name_bucket_id, _location_shm_id, _location_shm_offset); return NextCmdAddress(cmd); } CommandHeader header; uint32 program; uint32 name_bucket_id; uint32 location_shm_id; uint32 location_shm_offset; }; COMPILE_ASSERT(sizeof(GetUniformLocationBucket) == 20, Sizeof_GetUniformLocationBucket_is_not_24); COMPILE_ASSERT(offsetof(GetUniformLocationBucket, header) == 0, OffsetOf_GetUniformLocationBucket_header_not_0); COMPILE_ASSERT(offsetof(GetUniformLocationBucket, program) == 4, OffsetOf_GetUniformLocationBucket_program_not_4); COMPILE_ASSERT(offsetof(GetUniformLocationBucket, name_bucket_id) == 8, OffsetOf_GetUniformLocationBucket_name_bucket_id_not_8); COMPILE_ASSERT(offsetof(GetUniformLocationBucket, location_shm_id) == 12, OffsetOf_GetUniformLocationBucket_location_shm_id_not_12); COMPILE_ASSERT(offsetof(GetUniformLocationBucket, location_shm_offset) == 16, OffsetOf_GetUniformLocationBucket_location_shm_offset_not_16); struct InsertSyncPointCHROMIUM { typedef InsertSyncPointCHROMIUM ValueType; static const CommandId kCmdId = kInsertSyncPointCHROMIUM; static const cmd::ArgFlags kArgFlags = cmd::kFixed; CommandHeader header; }; #pragma pack(pop) } // namespace cmd } // namespace gles2 } // namespace gpu #endif // GPU_COMMAND_BUFFER_COMMON_GLES2_CMD_FORMAT_H_