web-dev-qa-db-ja.com

SDL2:高速ピクセル操作

特定のパラメーターの後で頻繁に変化するピクセルをモニターに描画したいと思います。例えば。赤と緑のピクセルが衝突すると、両方が消えるなどです。

すべてのフレームで、約100〜1000ピクセルを操作する必要があります。私はここでマルチスレッドアプローチを採用していますが、30FPS(私が欲しいもの)は得られません。現在、ピクセル配列をRAMに格納しています。これには、すべてのピクセルが含まれ、SDL_Surfaceがあります。配列内のピクセルが変更されると、Surfaceでも変更され、すべての操作が完了した後です。私の現在のアプローチは遅すぎるので、どうすれば速度を上げることができるかについて少し考えました。

私の現在の考えは次のとおりです。

  • OpenGLを使用してGPUで直接ピクセル操作を行うと、一部のフォーラムでは、「これはGPUの動作方法ではない」ため、現在のアプローチよりもはるかに遅いと言われています。
  • ピクセル配列を保存せず、BMP in RAMに直接保存し、それを操作してから、SDL_SurfaceまたはSDL_Textureに移動します。

Pixelをすばやく操作する方法について他にアプローチはありますか?

14
Nidhoegger

SDL_CreateTexture() w/_SDL_TEXTUREACCESS_STREAMING_ + SDL_UpdateTexture()は、適切なピクセル形式で十分に機能するようです。

デフォルトのレンダラーを使用している私のシステムでは:

_Renderer name: direct3d
Texture formats:
SDL_PIXELFORMAT_ARGB8888
SDL_PIXELFORMAT_YV12
SDL_PIXELFORMAT_IYUV
_

(ただし、opengl情報は同じです:)

_Renderer name: opengl
Texture formats:
SDL_PIXELFORMAT_ARGB8888
SDL_PIXELFORMAT_YV12
SDL_PIXELFORMAT_IYUV
_

_SDL_PIXELFORMAT_ARGB8888_は私に〜1ms /フレームを与えます:

_#include <SDL2/SDL.h>
#include <SDL2/SDL_render.h>
#include <iostream>
#include <vector>

using namespace std;

int main( int argc, char** argv )
{
    SDL_Init( SDL_INIT_EVERYTHING );
    atexit( SDL_Quit );

    SDL_Window* window = SDL_CreateWindow
        (
        "SDL2",
        SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED,
        600, 600,
        SDL_WINDOW_SHOWN
        );

    SDL_Renderer* renderer = SDL_CreateRenderer
        (
        window,
        -1,
        SDL_RENDERER_ACCELERATED
        );

    SDL_RendererInfo info;
    SDL_GetRendererInfo( renderer, &info );
    cout << "Renderer name: " << info.name << endl;
    cout << "Texture formats: " << endl;
    for( Uint32 i = 0; i < info.num_texture_formats; i++ )
    {
        cout << SDL_GetPixelFormatName( info.texture_formats[i] ) << endl;
    }

    const unsigned int texWidth = 1024;
    const unsigned int texHeight = 1024;
    SDL_Texture* texture = SDL_CreateTexture
        (
        renderer,
        SDL_PIXELFORMAT_ARGB8888,
        SDL_TEXTUREACCESS_STREAMING,
        texWidth, texHeight
        );

    vector< unsigned char > pixels( texWidth * texHeight * 4, 0 );

    SDL_Event event;
    bool running = true;
    while( running )
    {
        const Uint64 start = SDL_GetPerformanceCounter();

        SDL_SetRenderDrawColor( renderer, 0, 0, 0, SDL_ALPHA_OPAQUE );
        SDL_RenderClear( renderer );

        while( SDL_PollEvent( &event ) )
        {
            if( ( SDL_QUIT == event.type ) ||
                ( SDL_KEYDOWN == event.type && SDL_SCANCODE_ESCAPE == event.key.keysym.scancode ) )
            {
                running = false;
                break;
            }
        }

        // splat down some random pixels
        for( unsigned int i = 0; i < 1000; i++ )
        {
            const unsigned int x = Rand() % texWidth;
            const unsigned int y = Rand() % texHeight;

            const unsigned int offset = ( texWidth * 4 * y ) + x * 4;
            pixels[ offset + 0 ] = Rand() % 256;        // b
            pixels[ offset + 1 ] = Rand() % 256;        // g
            pixels[ offset + 2 ] = Rand() % 256;        // r
            pixels[ offset + 3 ] = SDL_ALPHA_OPAQUE;    // a
        }

        //unsigned char* lockedPixels;
        //int pitch;
        //SDL_LockTexture
        //    (
        //    texture,
        //    NULL,
        //    reinterpret_cast< void** >( &lockedPixels ),
        //    &pitch
        //    );
        //std::copy( pixels.begin(), pixels.end(), lockedPixels );
        //SDL_UnlockTexture( texture );

        SDL_UpdateTexture
            (
            texture,
            NULL,
            &pixels[0],
            texWidth * 4
            );

        SDL_RenderCopy( renderer, texture, NULL, NULL );
        SDL_RenderPresent( renderer );

        const Uint64 end = SDL_GetPerformanceCounter();
        const static Uint64 freq = SDL_GetPerformanceFrequency();
        const double seconds = ( end - start ) / static_cast< double >( freq );
        cout << "Frame time: " << seconds * 1000.0 << "ms" << endl;
    }

    SDL_DestroyRenderer( renderer );
    SDL_DestroyWindow( window );
    SDL_Quit();
}
_

Vsyncが有効になっていないことを確認してください(ドライバーで強制されている、コンポジターを実行しているなど)。そうでない場合、allフレーム時間は最大16ms(またはディスプレイの更新が設定されているもの)になります。 。

13
genpfault