/* 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 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) #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) #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, mCOLOR_ANY = -1 }; struct mImage { void* data; unsigned width; unsigned height; unsigned stride; unsigned depth; enum mColorFormat format; }; 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); uint32_t mColorConvert(uint32_t color, enum mColorFormat from, enum mColorFormat to); #ifndef PYCPARSE 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: return 1; case mCOLOR_ANY: break; } return 0; } 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; } #endif CXX_GUARD_END #endif