2023-03-20 10:43:44 +00:00
|
|
|
/* Copyright (c) 2013-2015 Jeffrey Pfau
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#ifndef M_IMAGE_H
|
|
|
|
#define M_IMAGE_H
|
|
|
|
|
|
|
|
#include <mgba-util/common.h>
|
|
|
|
|
|
|
|
CXX_GUARD_START
|
|
|
|
|
|
|
|
#ifdef COLOR_16_BIT
|
|
|
|
typedef uint16_t color_t;
|
|
|
|
#define BYTES_PER_PIXEL 2
|
|
|
|
#else
|
|
|
|
typedef uint32_t color_t;
|
|
|
|
#define BYTES_PER_PIXEL 4
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define M_R5(X) ((X) & 0x1F)
|
|
|
|
#define M_G5(X) (((X) >> 5) & 0x1F)
|
|
|
|
#define M_B5(X) (((X) >> 10) & 0x1F)
|
|
|
|
|
2023-03-22 08:42:27 +00:00
|
|
|
#define M_R8(X) ((M_R5(X) * 0x21) >> 2)
|
|
|
|
#define M_G8(X) ((M_G5(X) * 0x21) >> 2)
|
|
|
|
#define M_B8(X) ((M_B5(X) * 0x21) >> 2)
|
2023-03-20 10:43:44 +00:00
|
|
|
|
|
|
|
#define M_RGB5_TO_BGR8(X) ((M_R5(X) << 3) | (M_G5(X) << 11) | (M_B5(X) << 19))
|
|
|
|
#define M_RGB5_TO_RGB8(X) ((M_R5(X) << 19) | (M_G5(X) << 11) | (M_B5(X) << 3))
|
|
|
|
#define M_RGB8_TO_BGR5(X) ((((X) & 0xF8) >> 3) | (((X) & 0xF800) >> 6) | (((X) & 0xF80000) >> 9))
|
|
|
|
#define M_RGB8_TO_RGB5(X) ((((X) & 0xF8) << 7) | (((X) & 0xF800) >> 6) | (((X) & 0xF80000) >> 19))
|
|
|
|
|
|
|
|
#ifndef COLOR_16_BIT
|
|
|
|
#define M_COLOR_RED 0x000000FF
|
|
|
|
#define M_COLOR_GREEN 0x0000FF00
|
|
|
|
#define M_COLOR_BLUE 0x00FF0000
|
|
|
|
#define M_COLOR_ALPHA 0xFF000000
|
|
|
|
#define M_COLOR_WHITE 0x00FFFFFF
|
|
|
|
|
|
|
|
#define M_RGB8_TO_NATIVE(X) (((X) & 0x00FF00) | (((X) & 0x0000FF) << 16) | (((X) & 0xFF0000) >> 16))
|
|
|
|
#elif defined(COLOR_5_6_5)
|
|
|
|
#define M_COLOR_RED 0x001F
|
|
|
|
#define M_COLOR_GREEN 0x07E0
|
|
|
|
#define M_COLOR_BLUE 0xF800
|
|
|
|
#define M_COLOR_ALPHA 0x0000
|
|
|
|
#define M_COLOR_WHITE 0xFFDF
|
|
|
|
|
|
|
|
#define M_RGB8_TO_NATIVE(X) ((((X) & 0xF8) << 8) | (((X) & 0xFC00) >> 5) | (((X) & 0xF80000) >> 19))
|
|
|
|
#else
|
|
|
|
#define M_COLOR_RED 0x001F
|
|
|
|
#define M_COLOR_GREEN 0x03E0
|
|
|
|
#define M_COLOR_BLUE 0x7C00
|
|
|
|
#define M_COLOR_ALPHA 0x1000
|
|
|
|
#define M_COLOR_WHITE 0x7FFF
|
|
|
|
|
|
|
|
#define M_RGB8_TO_NATIVE(X) M_RGB8_TO_BGR5(X)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
enum mColorFormat {
|
|
|
|
mCOLOR_XBGR8 = 0x00001,
|
|
|
|
mCOLOR_XRGB8 = 0x00002,
|
|
|
|
mCOLOR_BGRX8 = 0x00004,
|
|
|
|
mCOLOR_RGBX8 = 0x00008,
|
|
|
|
mCOLOR_ABGR8 = 0x00010,
|
|
|
|
mCOLOR_ARGB8 = 0x00020,
|
|
|
|
mCOLOR_BGRA8 = 0x00040,
|
|
|
|
mCOLOR_RGBA8 = 0x00080,
|
|
|
|
mCOLOR_RGB5 = 0x00100,
|
|
|
|
mCOLOR_BGR5 = 0x00200,
|
|
|
|
mCOLOR_RGB565 = 0x00400,
|
|
|
|
mCOLOR_BGR565 = 0x00800,
|
|
|
|
mCOLOR_ARGB5 = 0x01000,
|
|
|
|
mCOLOR_ABGR5 = 0x02000,
|
|
|
|
mCOLOR_RGBA5 = 0x04000,
|
|
|
|
mCOLOR_BGRA5 = 0x08000,
|
|
|
|
mCOLOR_RGB8 = 0x10000,
|
|
|
|
mCOLOR_BGR8 = 0x20000,
|
|
|
|
mCOLOR_L8 = 0x40000,
|
2023-04-17 10:00:00 +00:00
|
|
|
mCOLOR_PAL8 = 0x80000,
|
2023-03-20 10:43:44 +00:00
|
|
|
|
|
|
|
mCOLOR_ANY = -1
|
|
|
|
};
|
|
|
|
|
2023-04-17 09:03:15 +00:00
|
|
|
#ifndef COLOR_16_BIT
|
|
|
|
#define mCOLOR_NATIVE mCOLOR_XBGR8
|
|
|
|
#elif !defined(COLOR_5_6_5)
|
|
|
|
#define mCOLOR_NATIVE mCOLOR_BGR5
|
|
|
|
#else
|
|
|
|
#define mCOLOR_NATIVE mCOLOR_RGB565
|
|
|
|
#endif
|
|
|
|
|
2023-03-22 08:42:27 +00:00
|
|
|
struct mImage {
|
|
|
|
void* data;
|
2023-04-17 10:00:00 +00:00
|
|
|
uint32_t* palette;
|
2023-03-22 08:42:27 +00:00
|
|
|
unsigned width;
|
|
|
|
unsigned height;
|
|
|
|
unsigned stride;
|
|
|
|
unsigned depth;
|
2023-04-17 10:00:00 +00:00
|
|
|
unsigned palSize;
|
2023-03-22 08:42:27 +00:00
|
|
|
enum mColorFormat format;
|
|
|
|
};
|
|
|
|
|
2023-07-24 04:43:15 +00:00
|
|
|
struct mPainter {
|
|
|
|
struct mImage* backing;
|
|
|
|
bool blend;
|
|
|
|
bool fill;
|
|
|
|
unsigned strokeWidth;
|
|
|
|
uint32_t strokeColor;
|
|
|
|
uint32_t fillColor;
|
|
|
|
};
|
|
|
|
|
2023-03-27 00:49:15 +00:00
|
|
|
struct VFile;
|
|
|
|
struct mImage* mImageCreate(unsigned width, unsigned height, enum mColorFormat format);
|
2023-04-02 09:59:11 +00:00
|
|
|
struct mImage* mImageCreateWithStride(unsigned width, unsigned height, unsigned stride, enum mColorFormat format);
|
|
|
|
struct mImage* mImageCreateFromConstBuffer(unsigned width, unsigned height, unsigned stride, enum mColorFormat format, const void* pixels);
|
2023-03-27 00:49:15 +00:00
|
|
|
struct mImage* mImageLoad(const char* path);
|
|
|
|
struct mImage* mImageLoadVF(struct VFile* vf);
|
2023-03-31 09:23:24 +00:00
|
|
|
struct mImage* mImageConvertToFormat(const struct mImage*, enum mColorFormat format);
|
2023-03-27 00:49:15 +00:00
|
|
|
void mImageDestroy(struct mImage*);
|
|
|
|
|
2023-03-31 09:42:01 +00:00
|
|
|
bool mImageSave(const struct mImage*, const char* path, const char* format);
|
|
|
|
bool mImageSaveVF(const struct mImage*, struct VFile* vf, const char* format);
|
2023-04-03 08:50:16 +00:00
|
|
|
|
2023-03-22 08:42:27 +00:00
|
|
|
uint32_t mImageGetPixel(const struct mImage* image, unsigned x, unsigned y);
|
|
|
|
uint32_t mImageGetPixelRaw(const struct mImage* image, unsigned x, unsigned y);
|
|
|
|
void mImageSetPixel(struct mImage* image, unsigned x, unsigned y, uint32_t color);
|
|
|
|
void mImageSetPixelRaw(struct mImage* image, unsigned x, unsigned y, uint32_t color);
|
|
|
|
|
2023-04-17 11:36:05 +00:00
|
|
|
void mImageSetPaletteSize(struct mImage* image, unsigned count);
|
|
|
|
void mImageSetPaletteEntry(struct mImage* image, unsigned index, uint32_t color);
|
|
|
|
|
2023-04-03 08:50:16 +00:00
|
|
|
void mImageBlit(struct mImage* image, const struct mImage* source, int x, int y);
|
2023-04-03 09:45:49 +00:00
|
|
|
void mImageComposite(struct mImage* image, const struct mImage* source, int x, int y);
|
|
|
|
void mImageCompositeWithAlpha(struct mImage* image, const struct mImage* source, int x, int y, float alpha);
|
2023-04-03 08:50:16 +00:00
|
|
|
|
2023-07-24 04:43:15 +00:00
|
|
|
void mPainterInit(struct mPainter*, struct mImage* backing);
|
|
|
|
void mPainterDrawRectangle(struct mPainter*, int x, int y, int width, int height);
|
2023-07-25 04:00:47 +00:00
|
|
|
void mPainterDrawLine(struct mPainter*, int x1, int y1, int x2, int y2);
|
2023-07-28 03:26:08 +00:00
|
|
|
void mPainterDrawCircle(struct mPainter*, int x, int y, int diameter);
|
2023-07-24 04:43:15 +00:00
|
|
|
|
2023-03-22 08:42:27 +00:00
|
|
|
uint32_t mColorConvert(uint32_t color, enum mColorFormat from, enum mColorFormat to);
|
2023-04-17 11:36:05 +00:00
|
|
|
uint32_t mImageColorConvert(uint32_t color, const struct mImage* from, enum mColorFormat to);
|
2023-03-22 08:42:27 +00:00
|
|
|
|
2023-03-20 10:43:44 +00:00
|
|
|
#ifndef PYCPARSE
|
2023-03-22 08:42:27 +00:00
|
|
|
static inline unsigned mColorFormatBytes(enum mColorFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case mCOLOR_XBGR8:
|
|
|
|
case mCOLOR_XRGB8:
|
|
|
|
case mCOLOR_BGRX8:
|
|
|
|
case mCOLOR_RGBX8:
|
|
|
|
case mCOLOR_ABGR8:
|
|
|
|
case mCOLOR_ARGB8:
|
|
|
|
case mCOLOR_BGRA8:
|
|
|
|
case mCOLOR_RGBA8:
|
|
|
|
return 4;
|
|
|
|
case mCOLOR_RGB5:
|
|
|
|
case mCOLOR_BGR5:
|
|
|
|
case mCOLOR_RGB565:
|
|
|
|
case mCOLOR_BGR565:
|
|
|
|
case mCOLOR_ARGB5:
|
|
|
|
case mCOLOR_ABGR5:
|
|
|
|
case mCOLOR_RGBA5:
|
|
|
|
case mCOLOR_BGRA5:
|
|
|
|
return 2;
|
|
|
|
case mCOLOR_RGB8:
|
|
|
|
case mCOLOR_BGR8:
|
|
|
|
return 3;
|
|
|
|
case mCOLOR_L8:
|
2023-04-17 10:00:00 +00:00
|
|
|
case mCOLOR_PAL8:
|
2023-03-22 08:42:27 +00:00
|
|
|
return 1;
|
|
|
|
case mCOLOR_ANY:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-03-31 09:42:01 +00:00
|
|
|
static inline bool mColorFormatHasAlpha(enum mColorFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case mCOLOR_XBGR8:
|
|
|
|
case mCOLOR_XRGB8:
|
|
|
|
case mCOLOR_BGRX8:
|
|
|
|
case mCOLOR_RGBX8:
|
|
|
|
case mCOLOR_RGB5:
|
|
|
|
case mCOLOR_BGR5:
|
|
|
|
case mCOLOR_RGB565:
|
|
|
|
case mCOLOR_BGR565:
|
|
|
|
case mCOLOR_RGB8:
|
|
|
|
case mCOLOR_BGR8:
|
|
|
|
case mCOLOR_L8:
|
|
|
|
return false;
|
|
|
|
case mCOLOR_ABGR8:
|
|
|
|
case mCOLOR_ARGB8:
|
|
|
|
case mCOLOR_BGRA8:
|
|
|
|
case mCOLOR_RGBA8:
|
|
|
|
case mCOLOR_ARGB5:
|
|
|
|
case mCOLOR_ABGR5:
|
|
|
|
case mCOLOR_RGBA5:
|
|
|
|
case mCOLOR_BGRA5:
|
2023-04-17 10:00:00 +00:00
|
|
|
case mCOLOR_PAL8:
|
2023-03-31 09:42:01 +00:00
|
|
|
return true;
|
|
|
|
case mCOLOR_ANY:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-03-20 10:43:44 +00:00
|
|
|
static inline color_t mColorFrom555(uint16_t value) {
|
|
|
|
#ifdef COLOR_16_BIT
|
|
|
|
#ifdef COLOR_5_6_5
|
|
|
|
color_t color = 0;
|
|
|
|
color |= (value & 0x001F) << 11;
|
|
|
|
color |= (value & 0x03E0) << 1;
|
|
|
|
color |= (value & 0x7C00) >> 10;
|
|
|
|
#else
|
|
|
|
color_t color = value;
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
color_t color = M_RGB5_TO_BGR8(value);
|
|
|
|
color |= (color >> 5) & 0x070707;
|
|
|
|
#endif
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
|
|
|
ATTRIBUTE_UNUSED static unsigned mColorMix5Bit(int weightA, unsigned colorA, int weightB, unsigned colorB) {
|
|
|
|
unsigned c = 0;
|
|
|
|
unsigned a, b;
|
|
|
|
#ifdef COLOR_16_BIT
|
|
|
|
#ifdef COLOR_5_6_5
|
|
|
|
a = colorA & 0xF81F;
|
|
|
|
b = colorB & 0xF81F;
|
|
|
|
a |= (colorA & 0x7C0) << 16;
|
|
|
|
b |= (colorB & 0x7C0) << 16;
|
|
|
|
c = ((a * weightA + b * weightB) / 16);
|
|
|
|
if (c & 0x08000000) {
|
|
|
|
c = (c & ~0x0FC00000) | 0x07C00000;
|
|
|
|
}
|
|
|
|
if (c & 0x0020) {
|
|
|
|
c = (c & ~0x003F) | 0x001F;
|
|
|
|
}
|
|
|
|
if (c & 0x10000) {
|
|
|
|
c = (c & ~0x1F800) | 0xF800;
|
|
|
|
}
|
|
|
|
c = (c & 0xF81F) | ((c >> 16) & 0x07C0);
|
|
|
|
#else
|
|
|
|
a = colorA & 0x7C1F;
|
|
|
|
b = colorB & 0x7C1F;
|
|
|
|
a |= (colorA & 0x3E0) << 16;
|
|
|
|
b |= (colorB & 0x3E0) << 16;
|
|
|
|
c = ((a * weightA + b * weightB) / 16);
|
|
|
|
if (c & 0x04000000) {
|
|
|
|
c = (c & ~0x07E00000) | 0x03E00000;
|
|
|
|
}
|
|
|
|
if (c & 0x0020) {
|
|
|
|
c = (c & ~0x003F) | 0x001F;
|
|
|
|
}
|
|
|
|
if (c & 0x8000) {
|
|
|
|
c = (c & ~0xF800) | 0x7C00;
|
|
|
|
}
|
|
|
|
c = (c & 0x7C1F) | ((c >> 16) & 0x03E0);
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
a = colorA & 0xFF;
|
|
|
|
b = colorB & 0xFF;
|
|
|
|
c |= ((a * weightA + b * weightB) / 16) & 0x1FF;
|
|
|
|
if (c & 0x00000100) {
|
|
|
|
c = 0x000000FF;
|
|
|
|
}
|
|
|
|
|
|
|
|
a = colorA & 0xFF00;
|
|
|
|
b = colorB & 0xFF00;
|
|
|
|
c |= ((a * weightA + b * weightB) / 16) & 0x1FF00;
|
|
|
|
if (c & 0x00010000) {
|
|
|
|
c = (c & 0x000000FF) | 0x0000FF00;
|
|
|
|
}
|
|
|
|
|
|
|
|
a = colorA & 0xFF0000;
|
|
|
|
b = colorB & 0xFF0000;
|
|
|
|
c |= ((a * weightA + b * weightB) / 16) & 0x1FF0000;
|
|
|
|
if (c & 0x01000000) {
|
|
|
|
c = (c & 0x0000FFFF) | 0x00FF0000;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return c;
|
|
|
|
}
|
2023-04-03 09:45:49 +00:00
|
|
|
|
|
|
|
ATTRIBUTE_UNUSED static uint32_t mColorMixARGB8(uint32_t colorA, uint32_t colorB) {
|
2023-07-23 19:37:59 +00:00
|
|
|
uint32_t alphaA = colorA >> 24;
|
|
|
|
if (!alphaA) {
|
2023-04-03 09:45:49 +00:00
|
|
|
return colorB;
|
|
|
|
}
|
2023-07-23 19:37:59 +00:00
|
|
|
uint32_t alphaB = colorB >> 24;
|
2023-04-03 09:45:49 +00:00
|
|
|
uint32_t color = 0;
|
2023-07-23 19:37:59 +00:00
|
|
|
|
|
|
|
#if 1
|
|
|
|
// TODO: Benchmark integer and float versions
|
2023-04-03 09:45:49 +00:00
|
|
|
uint32_t a, b;
|
2023-07-23 19:37:59 +00:00
|
|
|
uint32_t alpha = (alphaA * 0xFF) + alphaB * (0xFF - alphaA);
|
2023-04-03 09:45:49 +00:00
|
|
|
|
2023-07-23 19:37:59 +00:00
|
|
|
a = colorA & 0xFF;
|
|
|
|
a *= alphaA * 0xFF;
|
|
|
|
b = a;
|
2023-04-03 09:45:49 +00:00
|
|
|
|
2023-07-23 19:37:59 +00:00
|
|
|
a = colorB & 0xFF;
|
|
|
|
a *= alphaB * (0xFF - alphaA);
|
|
|
|
b += a;
|
|
|
|
|
|
|
|
b /= alpha;
|
|
|
|
if (b > 0xFF) {
|
2023-04-03 09:45:49 +00:00
|
|
|
color |= 0xFF;
|
2023-07-23 19:37:59 +00:00
|
|
|
} else {
|
|
|
|
color |= b;
|
2023-04-03 09:45:49 +00:00
|
|
|
}
|
2023-07-23 19:37:59 +00:00
|
|
|
|
|
|
|
a = (colorA >> 8) & 0xFF;
|
|
|
|
a *= alphaA * 0xFF;
|
|
|
|
b = a;
|
|
|
|
|
|
|
|
a = (colorB >> 8) & 0xFF;
|
|
|
|
a *= alphaB * (0xFF - alphaA);
|
|
|
|
b += a;
|
|
|
|
|
|
|
|
b /= alpha;
|
|
|
|
if (b > 0xFF) {
|
|
|
|
color |= 0xFF00;
|
|
|
|
} else {
|
|
|
|
color |= b << 8;
|
2023-04-03 09:45:49 +00:00
|
|
|
}
|
|
|
|
|
2023-07-23 19:37:59 +00:00
|
|
|
a = (colorA >> 16) & 0xFF;
|
|
|
|
a *= alphaA * 0xFF;
|
|
|
|
b = a;
|
2023-04-03 09:45:49 +00:00
|
|
|
|
2023-07-23 19:37:59 +00:00
|
|
|
a = (colorB >> 16) & 0xFF;
|
|
|
|
a *= alphaB * (0xFF - alphaA);
|
|
|
|
b += a;
|
2023-04-03 09:45:49 +00:00
|
|
|
|
2023-07-23 19:37:59 +00:00
|
|
|
b /= alpha;
|
|
|
|
if (b > 0xFF) {
|
|
|
|
color |= 0xFF0000;
|
|
|
|
} else {
|
|
|
|
color |= b << 16;
|
2023-04-03 09:45:49 +00:00
|
|
|
}
|
|
|
|
|
2023-07-23 19:37:59 +00:00
|
|
|
alpha /= 0xFF;
|
2023-04-03 09:45:49 +00:00
|
|
|
if (alpha > 0xFF) {
|
|
|
|
color |= 0xFF000000;
|
|
|
|
} else {
|
|
|
|
color |= alpha << 24;
|
|
|
|
}
|
2023-07-23 19:37:59 +00:00
|
|
|
#else
|
|
|
|
float ca, aa;
|
|
|
|
float cb, ab;
|
|
|
|
|
|
|
|
static const float r255 = 1 / 255.f;
|
|
|
|
aa = alphaA * r255;
|
|
|
|
ab = alphaB * r255;
|
|
|
|
float alpha = aa + ab * (1.f - aa);
|
|
|
|
float ralpha = 1.f / alpha;
|
|
|
|
alpha = alpha * 255.f;
|
|
|
|
color = ((int) alpha) << 24;
|
|
|
|
|
|
|
|
ca = ((colorA >> 16) & 0xFF) * r255;
|
|
|
|
cb = ((colorB >> 16) & 0xFF) * r255;
|
|
|
|
ca = ca * aa + cb * ab * (1.f - aa);
|
|
|
|
ca = ca * ralpha * 255.f;
|
|
|
|
if (ca > 255.f) {
|
|
|
|
ca = 255.f;
|
|
|
|
}
|
|
|
|
color |= ((int) ca) << 16;
|
|
|
|
|
|
|
|
ca = ((colorA >> 8) & 0xFF) * r255;
|
|
|
|
cb = ((colorB >> 8) & 0xFF) * r255;
|
|
|
|
ca = ca * aa + cb * ab * (1.f - aa);
|
|
|
|
ca = ca * ralpha * 255.f;
|
|
|
|
if (ca > 255.f) {
|
|
|
|
ca = 255.f;
|
|
|
|
}
|
|
|
|
color |= ((int) ca) << 8;
|
|
|
|
|
|
|
|
ca = (colorA & 0xFF) * r255;
|
|
|
|
cb = (colorB & 0xFF) * r255;
|
|
|
|
ca = ca * aa + cb * ab * (1.f - aa);
|
|
|
|
ca = ca * ralpha * 255.f;
|
|
|
|
if (ca > 255.f) {
|
|
|
|
ca = 255.f;
|
|
|
|
}
|
|
|
|
color |= (int) ca;
|
|
|
|
#endif
|
2023-04-03 09:45:49 +00:00
|
|
|
|
|
|
|
return color;
|
|
|
|
}
|
2023-03-20 10:43:44 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
CXX_GUARD_END
|
|
|
|
|
|
|
|
#endif
|