Ускорение производительности рендеринга текста openGL

Я использую библиотеку FreeType для извлечения каждой информации о глифе, такой как ширина, высота и растровое изображение. Это делается в функции инициализации, где меня не волнует, сколько времени это займет. Я храню информацию о каждом персонаже в контейнере карты, чтобы впоследствии я мог легко получить доступ к каждому персонажу.

Во время рендеринга я читаю строку и, используя строковый итератор, перебираю каждый символ и использую параметры глифа для создания полигонов, на которых я хочу нарисовать растровое изображение.

Полигоны нарисованы с использованием VAO вместе с VBO

Чтобы сделать его прозрачным, я использую смешивание, а для окрашивания используется шейдер.

Вот как выглядит моя функция рендеринга:

void CFreeType::RenderText(std::string text, int posX, int posY, Vector3d color)
{
ViewOrtho(RESx, RESy);

glEnable(GL_TEXTURE_2D);
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
glDisable(GL_DEPTH_TEST);
textShader->Use();
glUniform3dv(textColor_uniform_location, 1, color.v);

glBindVertexArray(VAO);
glBindBuffer(GL_ARRAY_BUFFER, VBO);

this->x = posX;
this->y = posY + FontHeight;

std::string::const_iterator c;
for(c = text.begin(); c != text.end(); c++)
{
//Load character from map
Character ch = Characters[*c];

//If we hit the '^' character, we probably want to change the text color
if(*c == '^')
{
//Check if the next character is a number, if so, change the text color and skip this character.
c++;
ch = Characters[*c];
if(isdigit(*c))
{
glUniform3dv(glGetUniformLocation(textShader->GetProgram(), "textColor"), 1, Color[*c-48].v); // *c-48 for conversion from ASCII to in int - '0' == 48
continue;
}
//In the other case go back to previous character ('^').
else
{
c--;
ch = Characters[*c];
}
}

//If we hit a new line character, move the new line below the previous and skip this character.
else if(*c == '\n')
{
x = posX;
y += FontHeight;
continue;
}

//If we hit tab character, insert 4 spaces
else if(*c == '\t')
{
x += (Characters[' '].AdvanceX >> 6) << 2; //Bit shifting is hopefuly a bit faster than multiplying/dividing.
continue;
}

xpos = x + ch.Bearing.x;
ypos = y - ch.Bearing.y;
font_width = ch.Size.x;
font_height = ch.Size.y;

float vertices[6][4] = {
{xpos, ypos, 0.0, 0.0},
{xpos, ypos + this->font_height, 0.0, 1.0},
{xpos + this->font_width, ypos + this->font_height, 1.0, 1.0},

{xpos, ypos, 0.0, 0.0},
{xpos + this->font_width, ypos + this->font_height, 1.0, 1.0},
{xpos + this->font_width, ypos, 1.0, 0.0}
};

//Render character
glBindTexture(GL_TEXTURE_2D, ch.TextureID);
glBufferSubData(GL_ARRAY_BUFFER, 0, sizeof(vertices), vertices);
glDrawArrays(GL_TRIANGLES, 0, 6);

x += (ch.AdvanceX >> 6);
}

glBindBuffer(GL_ARRAY_BUFFER, 0);
glBindVertexArray(0);

ViewPerspective();

textShader->StopShader();
glDisable(GL_BLEND);
glEnable(GL_DEPTH_TEST);
glDisable(GL_TEXTURE_2D);
}

Я думал об использовании списка отображения вместо VBO, но я не думаю, что это улучшит ситуацию.

С примерно 400 нарисованными символами я получаю только около 165 FPS, а без визуализации я получаю почти 390 FPS.

Буду признателен за любую помощь, ведущую к повышению производительности рендеринга текста.

1

Решение

Привязка текстуры и glDrawArrays() за персонажа? Не лучший способ сделать это.

Минимизируйте количество текстурных связей & рисовать звонки:

  1. Склейте все ваши глифы в (в идеале) единый текстурный атлас
  2. Бросьте всю информацию о вершине для текста фрейма в один VBO
  3. Нарисуйте все свои строки для кадра с одним glDrawArrays() вызов
3

Другие решения


По вопросам рекламы [email protected]