kopia lustrzana https://github.com/OpenRTX/OpenRTX
Adding support for 8bpp grayscale and 1bpp black/white pixel formats to SDL-based display emulator, keeping 16bpp RGB565 as the default one.
rodzic
fabc874d63
commit
b9517e5867
|
@ -62,7 +62,7 @@ linux_src = src + ['tests/platform/x64_uC.c',
|
||||||
'rtos/uC-CPU/Posix/cpu_c.c']
|
'rtos/uC-CPU/Posix/cpu_c.c']
|
||||||
|
|
||||||
|
|
||||||
linux_def = def + {'SCREEN_WIDTH': '160', 'SCREEN_HEIGHT': '128'}
|
linux_def = def + {'SCREEN_WIDTH': '160', 'SCREEN_HEIGHT': '128', 'PIX_FMT_RGB565': ''}
|
||||||
|
|
||||||
linux_inc = inc + ['rtos/uC-OS3/Ports/POSIX',
|
linux_inc = inc + ['rtos/uC-OS3/Ports/POSIX',
|
||||||
'rtos/uC-CPU/Posix']
|
'rtos/uC-CPU/Posix']
|
||||||
|
|
|
@ -34,13 +34,85 @@
|
||||||
* Screen dimensions, adjust basing on the size of the screen you need to
|
* Screen dimensions, adjust basing on the size of the screen you need to
|
||||||
* emulate
|
* emulate
|
||||||
*/
|
*/
|
||||||
|
#ifndef SCREEN_WIDTH
|
||||||
#define SCREEN_WIDTH 160
|
#define SCREEN_WIDTH 160
|
||||||
#define SCREEN_HEIGHT 128
|
#endif
|
||||||
|
|
||||||
|
#ifndef SCREEN_HEIGHT
|
||||||
|
#define SCREEN_HEIGHT 128
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(PIX_FMT_BW) && !defined(PIX_FMT_GRAYSC) && !defined(PIX_FMT_RGB565)
|
||||||
|
#warning No pixel format defined, defaulting to RGB565
|
||||||
|
#define PIX_FMT_RGB565
|
||||||
|
#endif
|
||||||
|
|
||||||
|
SDL_Window *window; /* SDL window */
|
||||||
|
SDL_Surface *renderSurface; /* SDL rendering surface */
|
||||||
|
void *frameBuffer; /* Pointer to framebuffer */
|
||||||
|
bool inProgress; /* Flag to signal when rendering is in progress */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
* Internal helper function which fetches pixel at position (x, y) from framebuffer
|
||||||
|
* and returns it in SDL-compatible format, which is ARGB8888.
|
||||||
|
*/
|
||||||
|
uint32_t fetchPixelFromFb(unsigned int x, unsigned int y)
|
||||||
|
{
|
||||||
|
uint32_t pixel = 0;
|
||||||
|
|
||||||
|
#ifdef PIX_FMT_BW
|
||||||
|
/*
|
||||||
|
* Black and white 1bpp format: framebuffer is an array of uint8_t, where
|
||||||
|
* each cell contains the values of eight pixels, one per bit.
|
||||||
|
*/
|
||||||
|
uint8_t *fb = (uint8_t *)(frameBuffer);
|
||||||
|
unsigned int cell = (x + y*SCREEN_WIDTH) / 8;
|
||||||
|
unsigned int elem = (x + y*SCREEN_WIDTH) % 8;
|
||||||
|
if(fb[cell] & (1 << elem)) pixel = 0xFFFFFFFF;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef PIX_FMT_GRAYSC
|
||||||
|
/*
|
||||||
|
* Convert from 8bpp grayscale to ARGB8888, we have to do nothing more that
|
||||||
|
* replicating the pixel value for the three components
|
||||||
|
*/
|
||||||
|
uint8_t *fb = (uint8_t *)(frameBuffer);
|
||||||
|
uint8_t px = fb[x + y*SCREEN_WIDTH];
|
||||||
|
|
||||||
|
pixel = 0xFF000000 | (px << 16) | (px << 8) | px;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef PIX_FMT_RGB565
|
||||||
|
/*
|
||||||
|
* SDL pixel format is ARGB8888, while ours is RGB565, thus we need
|
||||||
|
* to do some conversions when writing framebuffer content to the
|
||||||
|
* window. We also set alpha value to its maximum.
|
||||||
|
*/
|
||||||
|
uint16_t *fb = (uint16_t *)(frameBuffer);
|
||||||
|
|
||||||
|
uint32_t r = (fb[x + y*SCREEN_WIDTH] & 0xF800) >> 11;
|
||||||
|
uint32_t g = (fb[x + y*SCREEN_WIDTH] & 0x07E0) >> 5;
|
||||||
|
uint32_t b = (fb[x + y*SCREEN_WIDTH] & 0x001F) & 0x1F;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Here we do conversions by multiplying by some scaling factors,
|
||||||
|
* we use ints just because the precision of floats is not really
|
||||||
|
* needed.
|
||||||
|
* Conversion factors:
|
||||||
|
* - five bit to eight bit: 8.226
|
||||||
|
* - six bit to eight bit: 4.0476
|
||||||
|
*/
|
||||||
|
r = (r * 8) + (r * 226)/1000;
|
||||||
|
g = (g * 4) + (g * 476)/10000;
|
||||||
|
b = (b * 8) + (b * 226)/1000;
|
||||||
|
|
||||||
|
pixel = 0xFF000000 | (r << 16) | (g << 8) | b;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return pixel;
|
||||||
|
}
|
||||||
|
|
||||||
SDL_Window *window;
|
|
||||||
SDL_Surface *renderSurface;
|
|
||||||
uint16_t *frameBuffer;
|
|
||||||
bool inProgress;
|
|
||||||
|
|
||||||
void display_init()
|
void display_init()
|
||||||
{
|
{
|
||||||
|
@ -61,9 +133,35 @@ void display_init()
|
||||||
renderSurface = SDL_GetWindowSurface(window);
|
renderSurface = SDL_GetWindowSurface(window);
|
||||||
SDL_FillRect(renderSurface,NULL,0xFFFFFF);
|
SDL_FillRect(renderSurface,NULL,0xFFFFFF);
|
||||||
|
|
||||||
unsigned int scrSize = SCREEN_HEIGHT * SCREEN_WIDTH * sizeof(uint16_t);
|
/*
|
||||||
frameBuffer = (uint16_t *)(malloc(scrSize));
|
* Black and white pixel format: framebuffer type is uint8_t where each
|
||||||
memset(frameBuffer, 0xFFFF, scrSize);
|
* bit represents a pixel. We have to allocate
|
||||||
|
* (SCREEN_HEIGHT * SCREEN_WIDTH)/8 elements
|
||||||
|
*/
|
||||||
|
#ifdef PIX_FMT_BW
|
||||||
|
unsigned int fbSize = (SCREEN_HEIGHT * SCREEN_WIDTH)/8;
|
||||||
|
if((fbSize * 8) < (SCREEN_HEIGHT * SCREEN_WIDTH)) fbSize += 1; /* Compensate for eventual truncation error in division */
|
||||||
|
fbSize *= sizeof(uint8_t);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Grayscale pixel format: framebuffer type is uint8_t where each element
|
||||||
|
* controls one pixel
|
||||||
|
*/
|
||||||
|
#ifdef PIX_FMT_GRAYSC
|
||||||
|
unsigned int fbSize = SCREEN_HEIGHT * SCREEN_WIDTH * sizeof(uint8_t);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* RGB565 pixel format: framebuffer type is uint16_t where each element
|
||||||
|
* controls one pixel
|
||||||
|
*/
|
||||||
|
#ifdef PIX_FMT_RGB565
|
||||||
|
unsigned int fbSize = SCREEN_HEIGHT * SCREEN_WIDTH * sizeof(uint16_t);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
frameBuffer = malloc(fbSize);
|
||||||
|
memset(frameBuffer, 0xFFFF, fbSize);
|
||||||
inProgress = false;
|
inProgress = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -97,32 +195,11 @@ void display_renderRows(uint8_t startRow, uint8_t endRow)
|
||||||
Uint32 *pixels = (Uint32*)renderSurface->pixels;
|
Uint32 *pixels = (Uint32*)renderSurface->pixels;
|
||||||
inProgress = true;
|
inProgress = true;
|
||||||
|
|
||||||
for(int x = 0; x < SCREEN_WIDTH; x++)
|
for(unsigned int x = 0; x < SCREEN_WIDTH; x++)
|
||||||
{
|
{
|
||||||
for(int y = startRow; y < endRow; y++)
|
for(unsigned int y = startRow; y < endRow; y++)
|
||||||
{
|
{
|
||||||
/*
|
pixels[x + y*SCREEN_WIDTH] = fetchPixelFromFb(x, y);
|
||||||
* SDL pixel format is ARGB8888, while ours is RGB565, thus we need
|
|
||||||
* to do some conversions when writing framebuffer content to the
|
|
||||||
* window. We also set alpha value to its maximum.
|
|
||||||
*/
|
|
||||||
uint32_t r = (frameBuffer[x + y*SCREEN_WIDTH] & 0xF800) >> 11;
|
|
||||||
uint32_t g = (frameBuffer[x + y*SCREEN_WIDTH] & 0x07E0) >> 5;
|
|
||||||
uint32_t b = (frameBuffer[x + y*SCREEN_WIDTH] & 0x001F) & 0x1F;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Here we do conversions by multiplying by some scaling factors,
|
|
||||||
* we use ints just because the precision of floats is not really
|
|
||||||
* needed.
|
|
||||||
* Conversion factors:
|
|
||||||
* - five bit to eight bit: 8.226
|
|
||||||
* - six bit to eight bit: 4.0476
|
|
||||||
*/
|
|
||||||
r = (r * 8) + (r * 226)/1000;
|
|
||||||
g = (g * 4) + (g * 476)/10000;
|
|
||||||
b = (b * 8) + (b * 226)/1000;
|
|
||||||
|
|
||||||
pixels[x + y*SCREEN_WIDTH] = 0xFF000000 | (r << 16) | (g << 8) | b;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Ładowanie…
Reference in New Issue