特定のパラメーターの後で頻繁に変化するピクセルをモニターに描画したいと思います。例えば。赤と緑のピクセルが衝突すると、両方が消えるなどです。
すべてのフレームで、約100〜1000ピクセルを操作する必要があります。私はここでマルチスレッドアプローチを採用していますが、30FPS(私が欲しいもの)は得られません。現在、ピクセル配列をRAMに格納しています。これには、すべてのピクセルが含まれ、SDL_Surface
があります。配列内のピクセルが変更されると、Surfaceでも変更され、すべての操作が完了した後です。私の現在のアプローチは遅すぎるので、どうすれば速度を上げることができるかについて少し考えました。
私の現在の考えは次のとおりです。
SDL_Surface
またはSDL_Texture
に移動します。Pixelをすばやく操作する方法について他にアプローチはありますか?
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(またはディスプレイの更新が設定されているもの)になります。 。