summaryrefslogtreecommitdiffstats
path: root/gfx/thebes/gfxColor.h
diff options
context:
space:
mode:
Diffstat (limited to 'gfx/thebes/gfxColor.h')
-rw-r--r--gfx/thebes/gfxColor.h83
1 files changed, 83 insertions, 0 deletions
diff --git a/gfx/thebes/gfxColor.h b/gfx/thebes/gfxColor.h
new file mode 100644
index 000000000..4ad7d9f06
--- /dev/null
+++ b/gfx/thebes/gfxColor.h
@@ -0,0 +1,83 @@
+/* -*- 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 "mozilla/Attributes.h" // for MOZ_ALWAYS_INLINE
+#include "mozilla/EndianUtils.h" // for mozilla::NativeEndian::swapToBigEndian
+
+/**
+ * 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 = ((uint32_t*)from)[0], \
+ m1 = ((uint32_t*)from)[1], \
+ m2 = ((uint32_t*)from)[2], \
+ rgbr = mozilla::NativeEndian::swapToBigEndian(m0), \
+ gbrg = mozilla::NativeEndian::swapToBigEndian(m1), \
+ brgb = mozilla::NativeEndian::swapToBigEndian(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));
+ }
+}
+
+#endif /* _GFX_COLOR_H */