gecko-dev/gfx/thebes/gfxColor.h

256 lines
7.7 KiB
C

/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* 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 GFX_COLOR_H
#define GFX_COLOR_H
#include "gfxTypes.h"
#include "prbit.h" // for PR_ROTATE_(LEFT,RIGHT)32
#include "prio.h" // for ntohl
#include "mozilla/Attributes.h" // for MOZ_ALWAYS_INLINE
#define GFX_UINT32_FROM_BPTR(pbptr,i) (((uint32_t*)(pbptr))[i])
#if defined(IS_BIG_ENDIAN)
#define GFX_NTOHL(x) (x)
#define GFX_HAVE_CHEAP_NTOHL
#elif defined(_WIN32)
#if (_MSC_VER >= 1300) // also excludes MinGW
#include <stdlib.h>
#pragma intrinsic(_byteswap_ulong)
#define GFX_NTOHL(x) _byteswap_ulong(x)
#define GFX_HAVE_CHEAP_NTOHL
#else
// A reasonably fast generic little-endian implementation.
#define GFX_NTOHL(x) \
( (PR_ROTATE_RIGHT32((x),8) & 0xFF00FF00) | \
(PR_ROTATE_LEFT32((x),8) & 0x00FF00FF) )
#endif
#else
#define GFX_NTOHL(x) ntohl(x)
#define GFX_HAVE_CHEAP_NTOHL
#endif
/**
* GFX_0XFF_PPIXEL_FROM_BPTR(x)
*
* Avoid tortured construction of 32-bit ARGB pixel from 3 individual bytes
* of memory plus constant 0xFF. RGB bytes are already contiguous!
* Equivalent to: GFX_PACKED_PIXEL(0xff,r,g,b)
*
* Attempt to use fast byte-swapping instruction(s), e.g. bswap on x86, in
* preference to a sequence of shift/or operations.
*/
#if defined(GFX_HAVE_CHEAP_NTOHL)
#define GFX_0XFF_PPIXEL_FROM_UINT32(x) \
( (GFX_NTOHL(x) >> 8) | (0xFF << 24) )
#else
// A reasonably fast generic little-endian implementation.
#define GFX_0XFF_PPIXEL_FROM_UINT32(x) \
( (PR_ROTATE_LEFT32((x),16) | 0xFF00FF00) & ((x) | 0xFFFF00FF) )
#endif
#define GFX_0XFF_PPIXEL_FROM_BPTR(x) \
( GFX_0XFF_PPIXEL_FROM_UINT32(GFX_UINT32_FROM_BPTR((x),0)) )
/**
* GFX_BLOCK_RGB_TO_FRGB(from,to)
* sizeof(*from) == sizeof(char)
* sizeof(*to) == sizeof(uint32_t)
*
* Copy 4 pixels at a time, reading blocks of 12 bytes (RGB x4)
* and writing blocks of 16 bytes (FRGB x4)
*/
#define GFX_BLOCK_RGB_TO_FRGB(from,to) \
PR_BEGIN_MACRO \
uint32_t m0 = GFX_UINT32_FROM_BPTR(from,0), \
m1 = GFX_UINT32_FROM_BPTR(from,1), \
m2 = GFX_UINT32_FROM_BPTR(from,2), \
rgbr = GFX_NTOHL(m0), \
gbrg = GFX_NTOHL(m1), \
brgb = GFX_NTOHL(m2), \
p0, p1, p2, p3; \
p0 = 0xFF000000 | ((rgbr) >> 8); \
p1 = 0xFF000000 | ((rgbr) << 16) | ((gbrg) >> 16); \
p2 = 0xFF000000 | ((gbrg) << 8) | ((brgb) >> 24); \
p3 = 0xFF000000 | (brgb); \
to[0] = p0; to[1] = p1; to[2] = p2; to[3] = p3; \
PR_END_MACRO
/**
* Fast approximate division by 255. It has the property that
* for all 0 <= n <= 255*255, GFX_DIVIDE_BY_255(n) == n/255.
* But it only uses two adds and two shifts instead of an
* integer division (which is expensive on many processors).
*
* equivalent to ((v)/255)
*/
#define GFX_DIVIDE_BY_255(v) \
(((((unsigned)(v)) << 8) + ((unsigned)(v)) + 255) >> 16)
/**
* Fast premultiply
*
* equivalent to (((c)*(a))/255)
*/
uint8_t MOZ_ALWAYS_INLINE gfxPreMultiply(uint8_t c, uint8_t a) {
return GFX_DIVIDE_BY_255((c)*(a));
}
/**
* Pack the 4 8-bit channels (A,R,G,B)
* into a 32-bit packed NON-premultiplied pixel.
*/
uint32_t MOZ_ALWAYS_INLINE
gfxPackedPixelNoPreMultiply(uint8_t a, uint8_t r, uint8_t g, uint8_t b) {
return (((a) << 24) | ((r) << 16) | ((g) << 8) | (b));
}
/**
* Pack the 4 8-bit channels (A,R,G,B)
* into a 32-bit packed premultiplied pixel.
*/
uint32_t MOZ_ALWAYS_INLINE
gfxPackedPixel(uint8_t a, uint8_t r, uint8_t g, uint8_t b) {
if (a == 0x00)
return 0x00000000;
else if (a == 0xFF) {
return gfxPackedPixelNoPreMultiply(a, r, g, b);
} else {
return ((a) << 24) |
(gfxPreMultiply(r,a) << 16) |
(gfxPreMultiply(g,a) << 8) |
(gfxPreMultiply(b,a));
}
}
/**
* A color value, storing red, green, blue and alpha components.
* This class does not use premultiplied alpha.
*
* XXX should this use doubles (instead of gfxFloat), for consistency with
* cairo?
*/
struct THEBES_API gfxRGBA {
gfxFloat r, g, b, a;
enum PackedColorType {
PACKED_ABGR,
PACKED_ABGR_PREMULTIPLIED,
PACKED_ARGB,
PACKED_ARGB_PREMULTIPLIED,
PACKED_XRGB
};
gfxRGBA() { }
/**
* Intialize this color using explicit red, green, blue and alpha
* values.
*/
gfxRGBA(gfxFloat _r, gfxFloat _g, gfxFloat _b, gfxFloat _a=1.0) : r(_r), g(_g), b(_b), a(_a) {}
/**
* Initialize this color from a packed 32-bit color.
* The color value is interpreted based on colorType;
* all values use the native platform endianness.
*
* Resulting gfxRGBA stores non-premultiplied data.
*
* @see gfxRGBA::Packed
*/
gfxRGBA(uint32_t c, PackedColorType colorType = PACKED_ABGR) {
if (colorType == PACKED_ABGR ||
colorType == PACKED_ABGR_PREMULTIPLIED)
{
r = ((c >> 0) & 0xff) * (1.0 / 255.0);
g = ((c >> 8) & 0xff) * (1.0 / 255.0);
b = ((c >> 16) & 0xff) * (1.0 / 255.0);
a = ((c >> 24) & 0xff) * (1.0 / 255.0);
} else if (colorType == PACKED_ARGB ||
colorType == PACKED_XRGB ||
colorType == PACKED_ARGB_PREMULTIPLIED)
{
b = ((c >> 0) & 0xff) * (1.0 / 255.0);
g = ((c >> 8) & 0xff) * (1.0 / 255.0);
r = ((c >> 16) & 0xff) * (1.0 / 255.0);
a = ((c >> 24) & 0xff) * (1.0 / 255.0);
}
if (colorType == PACKED_ABGR_PREMULTIPLIED ||
colorType == PACKED_ARGB_PREMULTIPLIED)
{
if (a > 0.0) {
r /= a;
g /= a;
b /= a;
}
} else if (colorType == PACKED_XRGB) {
a = 1.0;
}
}
bool operator==(const gfxRGBA& other) const
{
return r == other.r && g == other.g && b == other.b && a == other.a;
}
bool operator!=(const gfxRGBA& other) const
{
return !(*this == other);
}
/**
* Returns this color value as a packed 32-bit integer. This reconstructs
* the int32_t based on the given colorType, always in the native byte order.
*
* Note: gcc 4.2.3 on at least Ubuntu (x86) does something strange with
* (uint8_t)(c * 255.0) << x, where the result is different than
* double d = c * 255.0; v = ((uint8_t) d) << x.
*/
uint32_t Packed(PackedColorType colorType = PACKED_ABGR) const {
gfxFloat rb = (r * 255.0);
gfxFloat gb = (g * 255.0);
gfxFloat bb = (b * 255.0);
gfxFloat ab = (a * 255.0);
if (colorType == PACKED_ABGR) {
return (uint8_t(ab) << 24) |
(uint8_t(bb) << 16) |
(uint8_t(gb) << 8) |
(uint8_t(rb) << 0);
}
if (colorType == PACKED_ARGB || colorType == PACKED_XRGB) {
return (uint8_t(ab) << 24) |
(uint8_t(rb) << 16) |
(uint8_t(gb) << 8) |
(uint8_t(bb) << 0);
}
rb *= a;
gb *= a;
bb *= a;
if (colorType == PACKED_ABGR_PREMULTIPLIED) {
return (((uint8_t)(ab) << 24) |
((uint8_t)(bb) << 16) |
((uint8_t)(gb) << 8) |
((uint8_t)(rb) << 0));
}
if (colorType == PACKED_ARGB_PREMULTIPLIED) {
return (((uint8_t)(ab) << 24) |
((uint8_t)(rb) << 16) |
((uint8_t)(gb) << 8) |
((uint8_t)(bb) << 0));
}
return 0;
}
};
#endif /* _GFX_COLOR_H */