-
Notifications
You must be signed in to change notification settings - Fork 17
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Support 16/24/32 bit color depth #70
Conversation
Each RGB565 pixel requires 2 bytes. Typically, when using 16 bits per pixel, these bits are split to represent red, green, and blue values packed into a 16-bit word, with RGB565 as the standard format. In this layout, 5 bits are allocated to red, 6 to green, and 5 to blue, with red occupying the most significant bits in framebuffer memory. However, determining whether these bytes are stored in big-endian or little-endian format in memory can be challenging. This means that, after arranging R, G, and B values into a 16-bit word in the application, you may need to swap the bytes if the endianness of the CPU differs from that of the display device. To convert 8-bit R, G, and B color values to a 16-bit RGB565 format for Mado, you can use a function like this: static uint16_t rgb888_to_rgb565(uint8_t r, uint8_t g, uint8_t b)
{
return (((uint16_t)r & 0xF8) << 8) | (((uint16_t)g & 0xFC) << 3) | (b >> 3);
} Pixman would be useful for providing the accelerated routines for such purpose. This conversion function provides color translation for Mado's framebuffer backend.
This pull request has been converted to a draft as further rework is required. |
In the Here is the code to obtain the result in the image. static void _twin_fbdev_put_span(twin_coord_t left,
twin_coord_t top,
twin_coord_t right,
twin_argb32_t *pixels,
void *closure)
{
....
twin_coord_t width = right - left;
off_t off = top * screen->width + left;
uint32_t *dest =
(uint32_t *) ((uintptr_t) tx->fb_base + (off * sizeof(*dest)));
+ /* If the framebuffer is in 16 bpp mode, convert each pixel from 32 bpp (ARGB) to 16 bpp (RGB565).
+ * - Extracts the red, green, and blue channels from the 32-bit color value.
+ * - Shifts and masks each channel to fit into the 16-bit RGB565 format:
+ * - 5 bits for red
+ * - 6 bits for green
+ * - 5 bits for blue
+ * - Combines the shifted color values into a single 16-bit pixel.
+ * If the framebuffer is in 32 bpp mode, directly copy the pixels to the framebuffer.
+ */
+ if(tx->fb_var.bits_per_pixel == 16){
+ for (int i = 0; i < width; i++) {
+ twin_argb32_t color = pixels[i];
+ uint8_t r = (color >> 16) & 0xFF;
+ uint8_t g = (color >> 8) & 0xFF;
+ uint8_t b = color & 0xFF;
+ uint16_t pixel_value = (((uint16_t)r & 0xF8) << 8) | (((uint16_t)g & 0xFC) << 3) | ((uint16_t)b >> 3);
+ dest[i] = pixel_value;
+ }
+ }else{
+ memcpy(dest, pixels, width * sizeof(*dest));
+ }
} Subsequently, I applied the code provided in #65 to fix the memory region in the framebuffer, resulting in the following output. static void _twin_fbdev_put_span(twin_coord_t left,
twin_coord_t top,
twin_coord_t right,
twin_argb32_t *pixels,
void *closure)
{
....
twin_coord_t width = right - left;
+ uint32_t *dest;
+ off_t off = sizeof(*dest) * left + top * tx->fb_fix.line_length;
+ dest = (uint32_t *) ((uintptr_t) tx->fb_base + off);
/* If the framebuffer is in 16 bpp mode, convert each pixel from 32 bpp (ARGB) to 16 bpp (RGB565).
* - Extracts the red, green, and blue channels from the 32-bit color value.
* - Shifts and masks each channel to fit into the 16-bit RGB565 format:
* - 5 bits for red
* - 6 bits for green
* - 5 bits for blue
* - Combines the shifted color values into a single 16-bit pixel.
* If the framebuffer is in 32 bpp mode, directly copy the pixels to the framebuffer.
*/
if(tx->fb_var.bits_per_pixel == 16){
for (int i = 0; i < width; i++) {
twin_argb32_t color = pixels[i];
uint8_t r = (color >> 16) & 0xFF;
uint8_t g = (color >> 8) & 0xFF;
uint8_t b = color & 0xFF;
uint16_t pixel_value = (((uint16_t)r & 0xF8) << 8) | (((uint16_t)g & 0xFC) << 3) | ((uint16_t)b >> 3);
dest[i] = pixel_value;
}
}else{
memcpy(dest, pixels, width * sizeof(*dest));
}
} |
This comment was marked as resolved.
This comment was marked as resolved.
This comment was marked as resolved.
This comment was marked as resolved.
Consider the use of helper macro. E.g., src/gfx/convert.h from DirectFB2. #define RGB32_TO_ARGB8565(pixel) ( 0xff0000 | \
(((pixel) & 0xf80000) >> 8) | \
(((pixel) & 0x00fc00) >> 5) | \
(((pixel) & 0x0000f8) >> 3) ) |
Determine the format of Linux framebuffer. See awtk-port/lcd_linux/fb_info.h from awtk. |
I use a helper macro from DirectFB2 to refactor the code, making it more concise. The results are shown in the figure. +#define RGB32_TO_ARGB8565(pixel)(((pixel & 0x00f80000) >> 8) | \
+ ((pixel & 0x0000fc00) >> 5) | \
+ ((pixel & 0x000000f8) >> 3) )
static void _twin_fbdev_put_span(twin_coord_t left,
twin_coord_t top,
twin_coord_t right,
twin_argb32_t *pixels,
void *closure)
{
....
twin_coord_t width = right - left;
uint32_t *dest;
off_t off = sizeof(*dest) * left + top * tx->fb_fix.line_length;
dest = (uint32_t *) ((uintptr_t) tx->fb_base + off);
/* If the framebuffer is in 16 bpp mode, convert each pixel from 32 bpp (ARGB) to 16 bpp (RGB565).
* - Extracts the red, green, and blue channels from the 32-bit color value.
* - Shifts and masks each channel to fit into the 16-bit RGB565 format:
* - 5 bits for red
* - 6 bits for green
* - 5 bits for blue
* - Combines the shifted color values into a single 16-bit pixel.
* If the framebuffer is in 32 bpp mode, directly copy the pixels to the framebuffer.
*/
if(tx->fb_var.bits_per_pixel == 16){
for (int i = 0; i < width; i++) {
+ dest[i] = RGB32_TO_ARGB8565(pixels[i]);
}
}else{
memcpy(dest, pixels, width * sizeof(*dest));
}
} |
After reviewing awtk-port/lcd_linux/fb_info.h from AWTK, I noticed that |
For the Linux framebuffer backend implementation, we shall at least distinguish the color format and apply the appropriate action or perform color format conversion. |
When implementing a function to check the color format, I referred to linux/drivers/video/pxa168fb.c from Marvell and compared it with awtk-port/lcd_linux/fb_info.h from AWTK. I noticed a difference in the The code is from Marvell. case PIX_FMT_RGB565:
var->bits_per_pixel = 16;
var->red.offset = 11; var->red.length = 5;
var->green.offset = 5; var->green.length = 6;
var->blue.offset = 0; var->blue.length = 5;
var->transp.offset = 0; var->transp.length = 0;
...
case PIX_FMT_RGBA888:
var->bits_per_pixel = 32;
var->red.offset = 16; var->red.length = 8;
var->green.offset = 8; var->green.length = 8;
var->blue.offset = 0; var->blue.length = 8;
var->transp.offset = 24; var->transp.length = 8;
break; The code is from AWTK. static inline bool_t fb_is_rgb565(fb_info_t* fb) {
struct fb_var_screeninfo* var = &(fb->var);
if (var->bits_per_pixel == 16 && var->red.offset == 0 && var->green.offset == 5 &&
var->blue.offset == 11 && var->red.length == 5 && var->green.length == 6 &&
var->blue.length == 5) {
return TRUE;
} else {
return FALSE;
}
}
static inline bool_t fb_is_rgba8888(fb_info_t* fb) {
struct fb_var_screeninfo* var = &(fb->var);
if (var->bits_per_pixel == 32 && var->red.offset == 0 && var->green.offset == 8 &&
var->blue.offset == 16 && var->red.length == 8 && var->green.length == 8 &&
var->blue.length == 8) {
return TRUE;
} else {
return FALSE;
}
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Refine the code changes.
I implemented a function to examine the Linux framebuffer format and tested it in 16 and 32 bpp environments to ensure it works correctly. ...
+static bool fb_is_rgb565(twin_fbdev_t* tx) {
+ return tx->fb_var.red.offset == 11 && tx->fb_var.red.length == 5 &&
+ tx->fb_var.green.offset == 5 && tx->fb_var.green.length == 6 &&
+ tx->fb_var.blue.offset == 0 && tx->fb_var.blue.length == 5;
+}
+
+static bool fb_is_argb32(twin_fbdev_t* tx) {
+ return tx->fb_var.red.offset == 16 && tx->fb_var.red.length == 8 &&
+ tx->fb_var.green.offset == 8 && tx->fb_var.green.length == 8 &&
+ tx->fb_var.blue.offset == 0 && tx->fb_var.blue.length == 8;
+}
static bool twin_fbdev_apply_config(twin_fbdev_t *tx)
{
/* Read changable information of the framebuffer */
if (ioctl(tx->fb_fd, FBIOGET_VSCREENINFO, &tx->fb_var) == -1) {
log_error("Failed to get framebuffer information");
return false;
}
/* Set the virtual screen size to be the same as the physical screen */
tx->fb_var.xres_virtual = tx->fb_var.xres;
tx->fb_var.yres_virtual = tx->fb_var.yres;
if (ioctl(tx->fb_fd, FBIOPUT_VSCREENINFO, &tx->fb_var) < 0) {
log_error("Failed to set framebuffer mode");
return false;
}
/* Read changable information of the framebuffer again */
if (ioctl(tx->fb_fd, FBIOGET_VSCREENINFO, &tx->fb_var) < 0) {
log_error("Failed to get framebuffer information");
return false;
}
+ /* Examine the framebuffer format */
+ if (tx->fb_var.bits_per_pixel == 16) {
+ if(!fb_is_rgb565(tx)){
+ log_error("Invalid framebuffer format for 16 bpp");
+ return false;
+ }
+ }else{
+ if(!fb_is_argb32(tx)){
+ log_error("Invalid framebuffer format for 32 bpp");
+ return false;
+ }
+ }
/* Read unchangable information of the framebuffer */
ioctl(tx->fb_fd, FBIOGET_FSCREENINFO, &tx->fb_fix);
/* Align the framebuffer memory address with the page size */
off_t pgsize = getpagesize();
off_t start = (off_t) tx->fb_fix.smem_start & (pgsize - 1);
/* Round up the framebuffer memory size to match the page size */
tx->fb_len = start + (size_t) tx->fb_fix.smem_len + (pgsize - 1);
tx->fb_len &= ~(pgsize - 1);
/* Map framebuffer device to the virtual memory */
tx->fb_base = mmap(NULL, tx->fb_len, PROT_READ | PROT_WRITE, MAP_SHARED,
tx->fb_fd, 0);
if (tx->fb_base == MAP_FAILED) {
log_error("Failed to mmap framebuffer");
return false;
}
return true;
} |
This comment was marked as resolved.
This comment was marked as resolved.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Rebase the latest main
branch.
47c0cc6
to
272be09
Compare
Consider the following (incomplete) macro-based code generator: /* color conversion */
#define ARGB32_TO_RGB565_PERLINE(dest, pixels, width) \
do { \
for (int i = 0; i < width; i++) \
dest[i] = ((pixels[i] & 0x00f80000) >> 8) | \
((pixels[i] & 0x0000fc00) >> 5) | \
((pixels[i] & 0x000000f8) >> 3); \
} while (0)
/* Requires validation in 24-bit per pixel environments. */
#define ARGB32_TO_RGB888_PERLINE(pixel) \
do { \
for (int i = 0; i < width; i++) \
dest[i] = 0xff000000 | pixels[i]; \
} while (0)
#define ARGB32_TO_ARGB32_PERLINE(pixel) memcpy(dest, pixels, width * sizeof(*dest))
#define FBDEV_PUT_SPAN_IMPL(bpp, op) \
static void _twin_fbdev_put_span##bpp( \
twin_coord_t left, twin_coord_t top, twin_coord_t right, \
twin_argb32_t *pixels, void *closure) \
{ \
uint32_t *dest; \
twin_screen_t *screen = SCREEN(closure); \
twin_fbdev_t *tx = PRIV(closure); \
off_t off = (top) * (screen->width) + (left); \
*(dest) = \
(uint32_t *) ((uintptr_t) tx->fb_base + (off * sizeof(uint32_t))); \
twin_coord_t width = right - left; \
op(dest, pixels, width); \
}
FBDEV_PUT_SPAN_IMPL(16, ARGB32_TO_RGB565_PERLINE)
FBDEV_PUT_SPAN_IMPL(24, ARGB32_TO_RGB888_PERLINE)
FBDEV_PUT_SPAN_IMPL(32, ARGB32_TO_ARGB32_PERLINE) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Rebase the latest main
branch and resolve the conflicts.
#define ARGB32_TO_RGB565_PERLINE(dest, pixels, width) \ | ||
do { \ | ||
for (int i = 0; i < width; i++) \ | ||
dest[i] = ((pixels[i] & 0x00f80000) >> 8) | \ | ||
((pixels[i] & 0x0000fc00) >> 5) | \ | ||
((pixels[i] & 0x000000f8) >> 3); \ | ||
} while (0) | ||
|
||
/* Requires validation in 24-bit per pixel environments */ | ||
#define ARGB32_TO_RGB888_PERLINE(dest, pixels, width) \ | ||
do { \ | ||
for (int i = 0; i < width; i++) \ | ||
dest[i] = 0xff000000 | pixels[i]; \ | ||
} while (0) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@Bennctu, can you review these color conversion macros and suggest any potential use of Pixman for performance? These macros handle ARGB32 to RGB565 and RGB888 conversions.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Pixman API does not support color conversion.
Mado currently supports 32-bit color depth for desktop and laptop display devices. To enhance its compatibility across diverse Linux framebuffer environments, the following changes have been introduced to support 16-bit, 24-bit, and 32-bit color depths: - Distinguish the color format and perform color format conversion. - Ensure the correctness of the framebuffer format based on the color format. Additionally, initial support for color format conversion in 24-bit per pixel environments has been implemented. However, this implementation requires further testing to verify its correctness and stability. These updates expand Mado's compatibility, enabling its use on devices with limited memory resources, such as the Raspberry Pi, while maintaining compatibility with desktop display devices.
Thank @p96114175 for contributing! |
Mado currently uses 32-bit bpp, supporting display devices on desktop/laptop computers, while Raspberry Pi devices use 16-bit bpp. To improve mado’s compatibility across various Linux framebuffer environments, testing showed that
tx->fb_var.bits_per_pixel automatically sets the appropriate bpp without requiring manual configuration.