2 * Copyright (c) 2018 Samsung Electronics Co., Ltd.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
18 #include <dali/internal/render/renderers/render-geometry.h>
21 #include <dali/internal/common/buffer-index.h>
22 #include <dali/internal/render/gl-resources/context.h>
23 #include <dali/internal/render/gl-resources/gpu-buffer.h>
24 #include <dali/internal/render/renderers/render-property-buffer.h>
25 #include <dali/internal/render/shaders/program.h>
37 mGeometryType( Dali::Geometry::TRIANGLES ),
38 mIndicesChanged(false),
39 mHasBeenUpdated(false),
40 mAttributesChanged(true)
48 void Geometry::GlContextCreated( Context& context )
52 void Geometry::GlContextDestroyed()
56 void Geometry::AddPropertyBuffer( Render::PropertyBuffer* propertyBuffer )
58 mVertexBuffers.PushBack( propertyBuffer );
59 mAttributesChanged = true;
62 void Geometry::SetIndexBuffer( Dali::Vector<uint16_t>& indices )
64 mIndices.Swap( indices );
65 mIndicesChanged = true;
68 void Geometry::RemovePropertyBuffer( const Render::PropertyBuffer* propertyBuffer )
70 const auto&& end = mVertexBuffers.End();
71 for( auto&& iter = mVertexBuffers.Begin(); iter != end; ++iter )
73 if( *iter == propertyBuffer )
75 //This will delete the gpu buffer associated to the RenderPropertyBuffer if there is one
76 mVertexBuffers.Remove( iter );
77 mAttributesChanged = true;
83 void Geometry::GetAttributeLocationFromProgram( Vector<GLint>& attributeLocation, Program& program, BufferIndex bufferIndex ) const
85 attributeLocation.Clear();
87 for( auto&& vertexBuffer : mVertexBuffers )
89 const uint32_t attributeCount = vertexBuffer->GetAttributeCount();
90 for( uint32_t j = 0; j < attributeCount; ++j )
92 const std::string& attributeName = vertexBuffer->GetAttributeName( j );
93 uint32_t index = program.RegisterCustomAttribute( attributeName );
94 GLint location = program.GetCustomAttributeLocation( index );
98 DALI_LOG_WARNING( "Attribute not found in the shader: %s\n", attributeName.c_str() );
101 attributeLocation.PushBack( location );
106 void Geometry::OnRenderFinished()
108 mHasBeenUpdated = false;
109 mAttributesChanged = false;
112 void Geometry::Upload( Context& context )
114 if( !mHasBeenUpdated )
117 if( mIndicesChanged )
119 if( mIndices.Empty() )
125 if ( mIndexBuffer == NULL )
127 mIndexBuffer = new GpuBuffer( context );
130 uint32_t bufferSize = static_cast<uint32_t>( sizeof( uint16_t ) * mIndices.Size() );
131 mIndexBuffer->UpdateDataBuffer( context, bufferSize, &mIndices[0], GpuBuffer::STATIC_DRAW, GpuBuffer::ELEMENT_ARRAY_BUFFER );
134 mIndicesChanged = false;
137 for( auto&& buffer : mVertexBuffers )
139 if( !buffer->Update( context ) )
141 //Vertex buffer is not ready ( Size, data or format has not been specified yet )
146 mHasBeenUpdated = true;
152 BufferIndex bufferIndex,
153 Vector<GLint>& attributeLocation,
154 uint32_t elementBufferOffset,
155 uint32_t elementBufferCount )
157 //Bind buffers to attribute locations
159 const uint32_t vertexBufferCount = static_cast<uint32_t>( mVertexBuffers.Count() );
160 for( uint32_t i = 0; i < vertexBufferCount; ++i )
162 mVertexBuffers[i]->BindBuffer( context, GpuBuffer::ARRAY_BUFFER );
163 base += mVertexBuffers[i]->EnableVertexAttributes( context, attributeLocation, base );
166 uint32_t numIndices(0u);
167 intptr_t firstIndexOffset(0u);
170 numIndices = static_cast<uint32_t>( mIndices.Size() );
172 if( elementBufferOffset != 0u )
174 elementBufferOffset = (elementBufferOffset >= numIndices ) ? numIndices - 1 : elementBufferOffset;
175 firstIndexOffset = elementBufferOffset * sizeof(GLushort);
176 numIndices -= elementBufferOffset;
179 if( elementBufferCount != 0u )
181 numIndices = std::min( elementBufferCount, numIndices );
185 GLenum geometryGLType(GL_NONE);
186 switch(mGeometryType)
188 case Dali::Geometry::TRIANGLES:
190 geometryGLType = GL_TRIANGLES;
193 case Dali::Geometry::LINES:
195 geometryGLType = GL_LINES;
198 case Dali::Geometry::POINTS:
200 geometryGLType = GL_POINTS;
203 case Dali::Geometry::TRIANGLE_STRIP:
205 geometryGLType = GL_TRIANGLE_STRIP;
208 case Dali::Geometry::TRIANGLE_FAN:
210 geometryGLType = GL_TRIANGLE_FAN;
213 case Dali::Geometry::LINE_LOOP:
215 geometryGLType = GL_LINE_LOOP;
218 case Dali::Geometry::LINE_STRIP:
220 geometryGLType = GL_LINE_STRIP;
226 if( mIndexBuffer && geometryGLType != GL_POINTS )
229 mIndexBuffer->Bind( context, GpuBuffer::ELEMENT_ARRAY_BUFFER );
230 // numIndices truncated, no value loss happening in practice
231 context.DrawElements( geometryGLType, static_cast<GLsizei>( numIndices ), GL_UNSIGNED_SHORT, reinterpret_cast<void*>( firstIndexOffset ) );
236 GLsizei numVertices(0u);
237 if( vertexBufferCount > 0 )
239 // truncated, no value loss happening in practice
240 numVertices = static_cast<GLsizei>( mVertexBuffers[0]->GetElementCount() );
243 context.DrawArrays( geometryGLType, 0, numVertices );
247 for( auto&& attribute : attributeLocation )
249 if( attribute != -1 )
251 context.DisableVertexAttributeArray( static_cast<GLuint>( attribute ) );
256 } // namespace SceneGraph
257 } // namespace Internal