Проблемы с рендерингом мозаики OpenGL с использованием glDrawArraysInstanced - PullRequest
0 голосов
/ 19 октября 2018

Я работаю над своей игрой с 2D-картой тайловых карт на C ++, использую несколько библиотек, таких как openGL, openAL и т. Д.

Сейчас я переделываю свою часть рендера в мою игру, чтобы улучшить ее с точки зрения производительности, прежде чемя начал работать над этим, у меня была рабочая версия, но она вызывала что-то вроде 180isch draw draw каждый кадр, что плохо.

Поэтому я решил пойти на рендеринг для каждой текстуры в моей игре (если яиспользовать 90 текстур на моей карте, я попытаюсь отрисовать 90 вызовов отрисовки, но если текстуры не будет на моем экране, я пропущу ее), когда я добрался до нее и попытался отрисовать что-то, я попал в угол, я сделал некоторые вычисления и вседолжно быть хорошо, но это не так.

if (game->data->currentWorld != 0)
{
    tileSizeX = (2.0f / tileXsideCount);
    tileSizeY = (2.0f / tileYsideCount);
    float worldXoffset = getWorldOffsetX();
    float worldYoffset = getWorldOffsetY();
    float playerXoffset = getPlayerOffsetX();
    float playerYoffset = getPlayerOffsetY();
    int tileXoffset = getTileOffsetX();
    int tileYoffset = getTileOffsetY();

    int* textRenderCount = new int[game->data->currentWorld->textureCount];

    float vertices[] =
    {
        -1 + tileSizeX, -1 + tileSizeY , 1.0f, 0.0f, 0.0f,  1.0f, 1.0f, // top right
        -1 + tileSizeX, -1, 1.0f, 0.0f, 0.0f,  1.0f, 0.0f, // bottom right
        -1, -1, 1.0f, 0.0f, 0.0f,  0.0f, 0.0f, // bottom left
        -1, -1 + tileSizeY, 1.0f, 0.0f, 0.0f,  0.0f, 1.0f  // top left
    };

    unsigned int indices[] = {
                    0, 1, 3, // first triangle
                    1, 2, 3  // second triangle
    };

    for (int i = 1; i < game->data->currentWorld->textureCount; i++)
    {
        glGenVertexArrays(1, &VAO);
        glGenBuffers(1, &VBO);
        glGenBuffers(1, &EBO);
        glBindVertexArray(VAO);
        glBindBuffer(GL_ARRAY_BUFFER, VBO);
        glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);


        glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, EBO);
        glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(indices), indices, GL_STATIC_DRAW);
        int offsetIndex = 0;

        unsigned int texture;

        if (game->data->loadedTextures[i] != NULL)
        {
            texture = game->data->loadedTextures[i];
        }
        else
        {
            std::cout << "Renderer::RenderLoop - texture not found for id: " << i << endl;
        }

        // position attribute
        glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 8 * sizeof(float), (void*)0);
        glEnableVertexAttribArray(0);
        // color attribute
        glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 8 * sizeof(float), (void*)(3 * sizeof(float)));
        glEnableVertexAttribArray(1);
        // texture coord attribute
        glVertexAttribPointer(2, 2, GL_FLOAT, GL_FALSE, 8 * sizeof(float), (void*)(6 * sizeof(float)));
        glEnableVertexAttribArray(2);

        tempShader->use();
        glBindTexture(GL_TEXTURE_2D, texture);

        for(int x = 0; x < game->data->currentWorld->sizeX; x++)
        {
            for (int y = 0; y < game->data->currentWorld->sizeY; y++)
            {
                if (
                    x > (tileXoffset - (tileXsideCount / 2) - 1) &&
                    x < (tileXoffset + (tileXsideCount)+1) &&
                    y >(tileYoffset - (tileYsideCount / 2) - 1) &&
                    y < (tileYoffset + (tileYsideCount)+1)
                    )
                {
                    if (game->data->currentWorld->world[x][y]->id == i)
                    {
                        if (game->data->debugTiles)
                        {
                            std::cout << "For tile: x: " << x << " y: " << y;
                        }
                        float offsetX = { ((x*1.0f) - worldXoffset)*tileSizeX};
                        float offsetY = { ((y*1.0f) - worldYoffset)*tileSizeY};
                        if (game->data->debugTiles)
                        {
                            std::cout << " offset is: " << offsetX << offsetY << endl;
                            std::cout << "worldXoffset is : " << worldXoffset << endl;
                            std::cout << "worldYoffset is : " << worldYoffset << endl;
                        }
                        stringstream ss;
                        std::string indexText;
                        ss << offsetIndex;
                        indexText = ss.str();
                        tempShader->setVec2(("offset[" + indexText + "]").c_str(), offsetX, offsetY);
                        offsetIndex++;
                    }
                }
            }
        }
        if (game->data->debugTiles)
        {
            std::cout << "RenderCount for last render request:" << offsetIndex+1 <<endl;
            game->data->debugTiles = false;
        }
    glBindVertexArray(VAO);
    glDrawArraysInstanced(GL_TRIANGLES,0, 6,offsetIndex+1);
    UnloadSceneBuffers();
    }

и вот мои функции для возврата смещений:

float Renderer::getWorldOffsetX()
{
    if (game->data->currentChar->xPos < tileXsideCount / 2)
    {
        return 0.0f;
    }
    else if(game->data->currentChar->xPos > game->data->currentWorld->sizeX - (tileXsideCount/2))
    {
        return (game->data->currentWorld->sizeX - (tileXsideCount / 2));
    }
    else
    {
        return (game->data->currentChar->xPos - tileXsideCount/2);
    }
};


float Renderer::getWorldOffsetY()
{
    if (game->data->currentChar->yPos < tileYsideCount / 2)
    {
        return 0.0f;
    }
    else if (game->data->currentChar->yPos > game->data->currentWorld->sizeY - (tileYsideCount / 2))
    {
        return (game->data->currentWorld->sizeY - (tileYsideCount / 2));
    }
    else
    {
        return (game->data->currentChar->yPos - tileYsideCount/2);
    }
};

float Renderer::getPlayerOffsetX()
{
    if (game->data->currentChar->xPos < tileXsideCount / 2)
    {
        return -1.0f + (game->data->currentChar->xPos*tileSizeX);
    }
    else if (game->data->currentChar->xPos > game->data->currentWorld->sizeX - (tileXsideCount / 2))
    {
        return +1.0f - (tileSizeX*((game->data->currentWorld->sizeX-tileXsideCount) - game->data->currentChar->xPos));
    }
    else
    {
        return 0.0f;
    }
};

float Renderer::getPlayerOffsetY()
{
    if (game->data->currentChar->yPos < tileYsideCount / 2)
    {
        return -1.0f + (game->data->currentChar->yPos*tileSizeY);
    }
    else if (game->data->currentChar->yPos > game->data->currentWorld->sizeY - (tileYsideCount / 2))
    {
        return +1.0f - (tileSizeY*((game->data->currentWorld->sizeY - tileYsideCount) - game->data->currentChar->yPos));
    }
    else
    {
        return 0.0f;
    }
};

int Renderer::getTileOffsetX()
{
    if (game->data->currentChar->xPos < tileXsideCount / 2)
    {
        return 0 + tileXsideCount/2;
    }
    else if (game->data->currentChar->xPos > game->data->currentWorld->sizeX - (tileXsideCount / 2))
    {
        return (game->data->currentWorld->sizeX-(tileXsideCount/2));
    }
    else
    {
        return game->data->currentChar->xPos;
    }
}

int Renderer::getTileOffsetY()
{
    if (game->data->currentChar->yPos < tileYsideCount / 2)
    {
        return 0 + tileYsideCount / 2;
}
    else if (game->data->currentChar->yPos > game->data->currentWorld->sizeY -(tileYsideCount / 2))
    {
        return (game->data->currentWorld->sizeY - (tileYsideCount / 2));
    }
    else
    {
        return game->data->currentChar->yPos;
    }
}

и вот мои шейдеры:

Vertex:

#version 330 core
layout(location = 0) in vec2 aPos;
layout(location = 1) in vec3 aColor;
layout(location = 2) in vec2 aTexCoord;

out vec2 TexCoord;

uniform vec2 offset[198];

void main()
{
    vec2 currentOffset = offset[gl_InstanceID];
    gl_Position = vec4(aPos + currentOffset,0.0f,1.0f);
    TexCoord = aTexCoord;
}

Фрагмент:

#version 330 core
out vec4 FragColor;

in vec3 ourColor;
in vec2 TexCoord;

uniform sampler2D ourTexture;

void main()
{
    FragColor = texture(ourTexture, TexCoord);
}

и это моя функция набора vec2 для шейдера (в основном она такая же, как на learnopengl.com):

void setVec2(const std::string &name, float input1,float input2)
{
    glUniform2f(glGetUniformLocation(ID, name.c_str()), input1,input2);
}

Редактировать: Эточто я вижу:

enter image description here

...