[我正在尝试作为Uint32 *像素数组访问SDL_Renderer的帧缓冲区,以传递给Libretro的显示功能。
我创建了这个最小的示例,以显示我在线看到的两种访问SDL帧缓冲区的方法,但两种方法都不起作用。
#include <iostream>
#include "SDL.h"
const int WIDTH = 50;
const int HEIGHT = 40;
int frameIndex = 0;
int main(int argc, char *argv[]) {
SDL_Init(SDL_INIT_VIDEO);
auto win = SDL_CreateWindow("Framebuffer test", SDL_WINDOWPOS_CENTERED | SDL_WINDOW_OPENGL, SDL_WINDOWPOS_CENTERED, WIDTH, HEIGHT, 0);
auto renderer = SDL_CreateRenderer(win, -1, SDL_RENDERER_ACCELERATED);
auto pixels = new uint32_t[WIDTH * HEIGHT]; // width x height x SDL_PIXELFORMAT_RGBA8888
auto surface = SDL_CreateRGBSurface(SDL_WINDOW_SHOWN, WIDTH, HEIGHT, 32, 0xff000000, 0x00ff0000, 0x0000ff00, 0x000000ff);
// Rect to draw on the screen
SDL_Rect rect;
rect.x = 0;
rect.y = 0;
rect.w = 10;
rect.h = 10;
while (true) {
SDL_Event e;
if (SDL_PollEvent(&e)) {
if (e.type == SDL_QUIT) {
break;
}
}
SDL_RenderClear(renderer);
SDL_SetRenderDrawColor(renderer, 255, 255, 255, 255);
SDL_RenderDrawRect(renderer, &rect);
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255);
// Access framebuffer: method 1
SDL_RenderReadPixels(renderer, nullptr, SDL_PIXELFORMAT_ARGB8888, pixels, 1);
auto surfacePixels = (Uint32*) surface->pixels;
for (int i = 0; i < WIDTH * HEIGHT; ++i)
surfacePixels[i] = pixels[i];
SDL_LockSurface(surface);
SDL_SaveBMP(surface, ("screen1_" + std::to_string(frameIndex) + ".bmp").c_str()); // Output buffer
SDL_UnlockSurface(surface);
// Access frame buffer: method 2
auto surf = SDL_GetWindowSurface(win);
SDL_LockSurface(surf);
SDL_SaveBMP(surf, ("screen2_" + std::to_string(frameIndex) + ".bmp").c_str()); // Output buffer
SDL_UnlockSurface(surf);
std::cout << "Screenshot: " << frameIndex++ << std::endl;
SDL_RenderPresent(renderer);
if (frameIndex >= 20)
break;
}
SDL_FreeSurface(surface);
SDL_DestroyRenderer(renderer);
SDL_DestroyWindow(win);
SDL_Quit();
return 0;
}
我期望看到黑屏显示为白色正方形,但是在第一种方法中,我看到了噪点,在第二种方法中,我看到了黑屏。我在做什么错?
您的SDL_RenderReadPixels音调参数为1。音调是单个图像行的字节长度。如果每个像素为4字节宽,而每一行为1字节...图像是否为1/4像素宽?听起来有问题。正确的音高是4 * WIDTH
(加上填充,但是32位格式的音高不高)。
其他说明:
SDL_CreateRGBSurface第一个参数是“标志-标志未使用,应设置为0”,而不是SDL_WINDOW_SHOWN
。没有任何变化,因为它被忽略,但至少会产生误导。
SDL_GetWindowSurface如果存在与窗口关联的渲染器,则无法使用。文档说:“您不能在此窗口中将此与3D或渲染API结合使用。”
SDL_LockSurface应该在访问表面像素之前,之后应解锁。在大多数情况下,不需要锁定,因此您看不到任何不同,但是在SaveBMP之前锁定在逻辑上是不正确的。