You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
339 lines
8.3 KiB
339 lines
8.3 KiB
/*
|
|
* Copyright 2022 Michael Drake <tlsa@netsurf-browser.org>
|
|
*
|
|
* This file is part of NetSurf, http://www.netsurf-browser.org/
|
|
*
|
|
* NetSurf is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; version 2 of the License.
|
|
*
|
|
* NetSurf is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/** \file
|
|
* Internal core bitmap interface.
|
|
*/
|
|
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
|
|
#include "utils/log.h"
|
|
#include "utils/errors.h"
|
|
|
|
#include "desktop/bitmap.h"
|
|
#include "desktop/gui_internal.h"
|
|
|
|
/** The client bitmap format. */
|
|
bitmap_fmt_t bitmap_fmt;
|
|
|
|
/** The client bitmap colour channel layout. */
|
|
struct bitmap_colour_layout bitmap_layout = {
|
|
.r = 0,
|
|
.g = 1,
|
|
.b = 2,
|
|
.a = 3,
|
|
};
|
|
|
|
/**
|
|
* Get the colour layout for the given bitmap format.
|
|
*
|
|
* \param[in] fmt Pixel format to get channel layout for,
|
|
* \return channel layout structure.
|
|
*/
|
|
static struct bitmap_colour_layout bitmap__get_colour_layout(
|
|
const bitmap_fmt_t *fmt)
|
|
{
|
|
switch (fmt->layout) {
|
|
default:
|
|
/* Fall through. */
|
|
case BITMAP_LAYOUT_R8G8B8A8:
|
|
return (struct bitmap_colour_layout) {
|
|
.r = 0,
|
|
.g = 1,
|
|
.b = 2,
|
|
.a = 3,
|
|
};
|
|
|
|
case BITMAP_LAYOUT_B8G8R8A8:
|
|
return (struct bitmap_colour_layout) {
|
|
.b = 0,
|
|
.g = 1,
|
|
.r = 2,
|
|
.a = 3,
|
|
};
|
|
|
|
case BITMAP_LAYOUT_A8R8G8B8:
|
|
return (struct bitmap_colour_layout) {
|
|
.a = 0,
|
|
.r = 1,
|
|
.g = 2,
|
|
.b = 3,
|
|
};
|
|
|
|
case BITMAP_LAYOUT_A8B8G8R8:
|
|
return (struct bitmap_colour_layout) {
|
|
.a = 0,
|
|
.b = 1,
|
|
.g = 2,
|
|
.r = 3,
|
|
};
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get string for given pixel layout.
|
|
*
|
|
* \param[in] layout The pixel layout to get string for,
|
|
* \return String for given layout.
|
|
*/
|
|
static const char *bitmap__layout_to_str(enum bitmap_layout layout)
|
|
{
|
|
const char *const str[] = {
|
|
[BITMAP_LAYOUT_R8G8B8A8] = "Byte-wise RGBA",
|
|
[BITMAP_LAYOUT_B8G8R8A8] = "Byte-wise BGRA",
|
|
[BITMAP_LAYOUT_A8R8G8B8] = "Byte-wise ARGB",
|
|
[BITMAP_LAYOUT_A8B8G8R8] = "Byte-wise ABGR",
|
|
[BITMAP_LAYOUT_RGBA8888] = "0xRRGGBBAA (native endian)",
|
|
[BITMAP_LAYOUT_BGRA8888] = "0xBBGGRRAA (native endian)",
|
|
[BITMAP_LAYOUT_ARGB8888] = "0xAARRGGBB (native endian)",
|
|
[BITMAP_LAYOUT_ABGR8888] = "0xAABBGGRR (native endian)",
|
|
};
|
|
|
|
if ((size_t)layout >= (sizeof(str)) / sizeof(*str) ||
|
|
str[layout] == NULL) {
|
|
return "Unknown";
|
|
}
|
|
|
|
return str[layout];
|
|
}
|
|
|
|
/* Exported function, documented in include/netsurf/bitmap.h */
|
|
void bitmap_set_format(const bitmap_fmt_t *bitmap_format)
|
|
{
|
|
bitmap_fmt = *bitmap_format;
|
|
|
|
NSLOG(netsurf, INFO, "Setting core bitmap format to: %s%s",
|
|
bitmap__layout_to_str(bitmap_format->layout),
|
|
bitmap_format->pma ? " pre multiplied alpha" : "");
|
|
|
|
bitmap_fmt.layout = bitmap_sanitise_bitmap_layout(bitmap_fmt.layout);
|
|
|
|
if (bitmap_format->layout != bitmap_fmt.layout) {
|
|
NSLOG(netsurf, INFO, "Sanitised layout to: %s",
|
|
bitmap__layout_to_str(bitmap_fmt.layout));
|
|
}
|
|
|
|
bitmap_layout = bitmap__get_colour_layout(&bitmap_fmt);
|
|
}
|
|
|
|
/**
|
|
* Swap colour component order.
|
|
*
|
|
* \param[in] width Bitmap width in pixels.
|
|
* \param[in] height Bitmap height in pixels.
|
|
* \param[in] buffer Pixel buffer.
|
|
* \param[in] rowstride Pixel buffer row stride in bytes.
|
|
* \param[in] to Pixel layout to convert to.
|
|
* \param[in] from Pixel layout to convert from.
|
|
*/
|
|
static inline void bitmap__format_convert(
|
|
int width,
|
|
int height,
|
|
uint8_t *buffer,
|
|
size_t rowstride,
|
|
struct bitmap_colour_layout to,
|
|
struct bitmap_colour_layout from)
|
|
{
|
|
/* Just swapping the components around */
|
|
for (int y = 0; y < height; y++) {
|
|
uint8_t *row = buffer;
|
|
|
|
for (int x = 0; x < width; x++) {
|
|
const uint32_t px = *((uint32_t *)(void *) row);
|
|
|
|
row[to.r] = ((const uint8_t *) &px)[from.r];
|
|
row[to.g] = ((const uint8_t *) &px)[from.g];
|
|
row[to.b] = ((const uint8_t *) &px)[from.b];
|
|
row[to.a] = ((const uint8_t *) &px)[from.a];
|
|
|
|
row += sizeof(uint32_t);
|
|
}
|
|
|
|
buffer += rowstride;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convert plain alpha to premultiplied alpha.
|
|
*
|
|
* \param[in] width Bitmap width in pixels.
|
|
* \param[in] height Bitmap height in pixels.
|
|
* \param[in] buffer Pixel buffer.
|
|
* \param[in] rowstride Pixel buffer row stride in bytes.
|
|
* \param[in] to Pixel layout to convert to.
|
|
* \param[in] from Pixel layout to convert from.
|
|
*/
|
|
static inline void bitmap__format_convert_to_pma(
|
|
int width,
|
|
int height,
|
|
uint8_t *buffer,
|
|
size_t rowstride,
|
|
struct bitmap_colour_layout to,
|
|
struct bitmap_colour_layout from)
|
|
{
|
|
for (int y = 0; y < height; y++) {
|
|
uint8_t *row = buffer;
|
|
|
|
for (int x = 0; x < width; x++) {
|
|
const uint32_t px = *((uint32_t *)(void *) row);
|
|
uint32_t a, r, g, b;
|
|
|
|
r = ((const uint8_t *) &px)[from.r];
|
|
g = ((const uint8_t *) &px)[from.g];
|
|
b = ((const uint8_t *) &px)[from.b];
|
|
a = ((const uint8_t *) &px)[from.a];
|
|
|
|
if (a != 0) {
|
|
r = ((r * (a + 1)) >> 8) & 0xff;
|
|
g = ((g * (a + 1)) >> 8) & 0xff;
|
|
b = ((b * (a + 1)) >> 8) & 0xff;
|
|
} else {
|
|
r = g = b = 0;
|
|
}
|
|
|
|
row[to.r] = r;
|
|
row[to.g] = g;
|
|
row[to.b] = b;
|
|
row[to.a] = a;
|
|
|
|
row += sizeof(uint32_t);
|
|
}
|
|
|
|
buffer += rowstride;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convert from premultiplied alpha to plain alpha.
|
|
*
|
|
* \param[in] width Bitmap width in pixels.
|
|
* \param[in] height Bitmap height in pixels.
|
|
* \param[in] buffer Pixel buffer.
|
|
* \param[in] rowstride Pixel buffer row stride in bytes.
|
|
* \param[in] to Pixel layout to convert to.
|
|
* \param[in] from Pixel layout to convert from.
|
|
*/
|
|
static inline void bitmap__format_convert_from_pma(
|
|
int width,
|
|
int height,
|
|
uint8_t *buffer,
|
|
size_t rowstride,
|
|
struct bitmap_colour_layout to,
|
|
struct bitmap_colour_layout from)
|
|
{
|
|
for (int y = 0; y < height; y++) {
|
|
uint8_t *row = buffer;
|
|
|
|
for (int x = 0; x < width; x++) {
|
|
const uint32_t px = *((uint32_t *)(void *) row);
|
|
uint32_t a, r, g, b;
|
|
|
|
r = ((const uint8_t *) &px)[from.r];
|
|
g = ((const uint8_t *) &px)[from.g];
|
|
b = ((const uint8_t *) &px)[from.b];
|
|
a = ((const uint8_t *) &px)[from.a];
|
|
|
|
if (a != 0) {
|
|
r = (r << 8) / a;
|
|
g = (g << 8) / a;
|
|
b = (b << 8) / a;
|
|
|
|
r = (r > 255) ? 255 : r;
|
|
g = (g > 255) ? 255 : g;
|
|
b = (b > 255) ? 255 : b;
|
|
} else {
|
|
r = g = b = 0;
|
|
}
|
|
|
|
row[to.r] = r;
|
|
row[to.g] = g;
|
|
row[to.b] = b;
|
|
row[to.a] = a;
|
|
|
|
row += sizeof(uint32_t);
|
|
}
|
|
|
|
buffer += rowstride;
|
|
}
|
|
}
|
|
|
|
/* Exported function, documented in desktop/bitmap.h */
|
|
void bitmap_format_convert(void *bitmap,
|
|
const bitmap_fmt_t *fmt_from,
|
|
const bitmap_fmt_t *fmt_to)
|
|
{
|
|
int width = guit->bitmap->get_width(bitmap);
|
|
int height = guit->bitmap->get_height(bitmap);
|
|
bool opaque = guit->bitmap->get_opaque(bitmap);
|
|
uint8_t *buffer = guit->bitmap->get_buffer(bitmap);
|
|
size_t rowstride = guit->bitmap->get_rowstride(bitmap);
|
|
struct bitmap_colour_layout to = bitmap__get_colour_layout(fmt_to);
|
|
struct bitmap_colour_layout from = bitmap__get_colour_layout(fmt_from);
|
|
|
|
NSLOG(netsurf, DEEPDEBUG, "%p: format conversion (%u%s --> %u%s)",
|
|
bitmap,
|
|
fmt_from->layout, fmt_from->pma ? " pma" : "",
|
|
fmt_to->layout, fmt_to->pma ? " pma" : "");
|
|
|
|
if (fmt_from->pma == fmt_to->pma) {
|
|
/* Just component order to switch. */
|
|
bitmap__format_convert(
|
|
width, height, buffer,
|
|
rowstride, to, from);
|
|
|
|
} else if (opaque == false) {
|
|
/* Need to do conversion to/from premultiplied alpha. */
|
|
if (fmt_to->pma) {
|
|
bitmap__format_convert_to_pma(
|
|
width, height, buffer,
|
|
rowstride, to, from);
|
|
} else {
|
|
bitmap__format_convert_from_pma(
|
|
width, height, buffer,
|
|
rowstride, to, from);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Exported function, documented in desktop/bitmap.h */
|
|
bool bitmap_test_opaque(void *bitmap)
|
|
{
|
|
int width = guit->bitmap->get_width(bitmap);
|
|
int height = guit->bitmap->get_height(bitmap);
|
|
size_t rowstride = guit->bitmap->get_rowstride(bitmap);
|
|
const uint8_t *buffer = guit->bitmap->get_buffer(bitmap);
|
|
|
|
width *= sizeof(uint32_t);
|
|
|
|
for (int y = 0; y < height; y++) {
|
|
const uint8_t *row = buffer;
|
|
|
|
for (int x = bitmap_layout.a; x < width; x += 4) {
|
|
if (row[x] != 0xff) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
buffer += rowstride;
|
|
}
|
|
|
|
return true;
|
|
}
|