59 wxString vendor( glGetString( GL_VENDOR ) );
65 if( vendor.StartsWith(
"Intel" ) || vendor.Contains(
"etnaviv" ) )
75 glBindBuffer( GL_ARRAY_BUFFER, 0 );
76 checkGlError(
"allocating video memory for cached container", __FILE__, __LINE__ );
96 throw std::runtime_error(
"OpenGL no longer available!" );
99 m_vertices =
static_cast<VERTEX*
>( glMapBuffer( GL_ARRAY_BUFFER, GL_READ_WRITE ) );
101 if(
checkGlError(
"mapping vertices buffer", __FILE__, __LINE__ ) == GL_NO_ERROR
109 glBindBuffer( GL_ARRAY_BUFFER, 0 );
110 throw std::runtime_error(
"Could not map vertex buffer: glMapBuffer returned null" );
123 glUnmapBuffer( GL_ARRAY_BUFFER );
124 checkGlError(
"unmapping vertices buffer", __FILE__, __LINE__ );
125 glBindBuffer( GL_ARRAY_BUFFER, 0 );
127 checkGlError(
"unbinding vertices buffer", __FILE__, __LINE__ );
129 catch(
const std::runtime_error& err )
131 wxLogError( wxT(
"OpenGL did not shut down properly.\n\n%s" ), err.what() );
146 wxT(
"Resizing & defragmenting container from %d to %d" ),
m_currentSize,
153#ifdef KICAD_GAL_PROFILE
160 glUnmapBuffer( GL_ARRAY_BUFFER );
163 glGenBuffers( 1, &newBuffer );
167#ifdef KICAD_GAL_PROFILE
169 glGetIntegerv( GL_ELEMENT_ARRAY_BUFFER_BINDING, &eaBuffer );
170 wxASSERT( eaBuffer == 0 );
172 glBindBuffer( GL_ELEMENT_ARRAY_BUFFER, newBuffer );
173 glBufferData( GL_ELEMENT_ARRAY_BUFFER, aNewSize *
VERTEX_SIZE,
nullptr, GL_DYNAMIC_DRAW );
174 checkGlError(
"creating buffer during defragmentation", __FILE__, __LINE__ );
176 ITEMS::iterator it, it_end;
180 for( it =
m_items.begin(), it_end =
m_items.end(); it != it_end; ++it )
184 int itemSize = item->
GetSize();
187 glCopyBufferSubData( GL_ARRAY_BUFFER, GL_ELEMENT_ARRAY_BUFFER, itemOffset *
VERTEX_SIZE,
194 newOffset += itemSize;
198 if(
m_item->GetSize() > 0 )
200 glCopyBufferSubData( GL_ARRAY_BUFFER, GL_ELEMENT_ARRAY_BUFFER,
204 m_item->setOffset( newOffset );
209 glBindBuffer( GL_ELEMENT_ARRAY_BUFFER, 0 );
210 glBindBuffer( GL_ARRAY_BUFFER, 0 );
224 catch(
const std::runtime_error& )
231 checkGlError(
"switching buffers during defragmentation", __FILE__, __LINE__ );
233#ifdef KICAD_GAL_PROFILE
258 wxT(
"Resizing & defragmenting container (memcpy) from %d to %d" ),
m_currentSize,
265#ifdef KICAD_GAL_PROFILE
273 glGenBuffers( 1, &newBuffer );
277#ifdef KICAD_GAL_PROFILE
279 glGetIntegerv( GL_ELEMENT_ARRAY_BUFFER_BINDING, &eaBuffer );
280 wxASSERT( eaBuffer == 0 );
283 glBindBuffer( GL_ELEMENT_ARRAY_BUFFER, newBuffer );
284 glBufferData( GL_ELEMENT_ARRAY_BUFFER, aNewSize *
VERTEX_SIZE,
nullptr, GL_DYNAMIC_DRAW );
285 newBufferMem =
static_cast<VERTEX*
>( glMapBuffer( GL_ELEMENT_ARRAY_BUFFER, GL_WRITE_ONLY ) );
286 checkGlError(
"creating buffer during defragmentation", __FILE__, __LINE__ );
288 if( newBufferMem ==
nullptr )
290 glBindBuffer( GL_ELEMENT_ARRAY_BUFFER, 0 );
291 glDeleteBuffers( 1, &newBuffer );
298 glUnmapBuffer( GL_ELEMENT_ARRAY_BUFFER );
299 glBindBuffer( GL_ELEMENT_ARRAY_BUFFER, 0 );
310 catch(
const std::runtime_error& )
317 checkGlError(
"switching buffers during defragmentation", __FILE__, __LINE__ );
319#ifdef KICAD_GAL_PROFILE
341 unsigned int size = 0;
343 for(
const auto& item :
m_items )
345 size += item->GetSize();
void Map() override
Finish the vertices updates stage.
bool m_isMapped
Vertex buffer handle.
virtual unsigned int AllItemsSize() const override
bool IsMapped() const override
Prepare the container for vertices updates.
bool defragmentResizeMemcpy(unsigned int aNewSize)
Flag saying if vertex buffer is currently mapped.
void Unmap() override
Finish the vertices updates stage.
CACHED_CONTAINER_GPU(unsigned int aSize=DEFAULT_SIZE)
bool defragmentResize(unsigned int aNewSize) override
Remove empty spaces between chunks and optionally resizes the container.
unsigned int m_glBufferHandle
Flag saying whether it is safe to use glCopyBufferSubData.
unsigned int m_chunkOffset
Maximal vertex index number stored in the container.
VERTEX_ITEM * m_item
Properties of currently modified chunk & item.
FREE_CHUNK_MAP m_freeChunks
Stored VERTEX_ITEMs.
void defragment(VERTEX *aTarget)
Transfer all stored data to a new buffer, removing empty spaces between the data chunks in the contai...
CACHED_CONTAINER(unsigned int aSize=DEFAULT_SIZE)
ITEMS m_items
Currently modified item.
unsigned int m_currentSize
Store the initial size, so it can be resized to this on Clear()
unsigned int m_freeSpace
Current container size, expressed in vertices.
unsigned int usedSpace() const
Return size of the used memory space.
void setOffset(unsigned int aOffset)
Set data offset in the container.
unsigned int GetOffset() const
Return data offset in the container.
unsigned int GetSize() const
Return information about number of vertices stored.
A small class to help profiling.
void Stop()
Save the time when this function was called, and set the counter stane to stop.
double msecs(bool aSinceLast=false)
static const wxChar *const traceGalCachedContainerGpu
Flag to enable debug output of the GAL OpenGL GPU cached container.
const wxChar *const traceGalProfile
Flag to enable debug output of GAL performance profiling.
The Cairo implementation of the graphics abstraction layer.
static constexpr size_t VERTEX_SIZE
wxLogTrace helper definitions.
#define KI_TRACE(aWhat,...)
int checkGlError(const std::string &aInfo, const char *aFile, int aLine, bool aThrow)
Check if a recent OpenGL operation has failed.
Class to handle an item held in a container.