blob: aae6ba5f116a2d121719621ebee7152dedb9c2b5 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001/*
2 * Copyright 2006 The Android Open Source Project
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
reed@android.com8a1c16f2008-12-17 15:59:43 +00008#include "SkAvoidXfermode.h"
9#include "SkColorPriv.h"
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000010#include "SkReadBuffer.h"
11#include "SkWriteBuffer.h"
robertphillips@google.comb83b6b42013-01-22 14:32:09 +000012#include "SkString.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000013
reed@google.com30da7452012-12-17 19:55:24 +000014SkAvoidXfermode::SkAvoidXfermode(SkColor opColor, U8CPU tolerance, Mode mode) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000015 if (tolerance > 255) {
16 tolerance = 255;
17 }
reed9fa60da2014-08-21 07:59:51 -070018 fTolerance = SkToU8(tolerance);
reed@android.com8a1c16f2008-12-17 15:59:43 +000019 fOpColor = opColor;
20 fDistMul = (256 << 14) / (tolerance + 1);
21 fMode = mode;
22}
23
reed9fa60da2014-08-21 07:59:51 -070024#ifdef SK_SUPPORT_LEGACY_DEEPFLATTENING
25SkAvoidXfermode::SkAvoidXfermode(SkReadBuffer& buffer) : INHERITED(buffer) {
djsollen@google.comc73dd5c2012-08-07 15:54:32 +000026 fOpColor = buffer.readColor();
27 fDistMul = buffer.readUInt();
28 fMode = (Mode)buffer.readUInt();
reed@android.com8a1c16f2008-12-17 15:59:43 +000029}
reed9fa60da2014-08-21 07:59:51 -070030#endif
31
32SkFlattenable* SkAvoidXfermode::CreateProc(SkReadBuffer& buffer) {
33 const SkColor color = buffer.readColor();
34 const unsigned tolerance = buffer.readUInt();
35 const unsigned mode = buffer.readUInt();
36 return Create(color, tolerance, (Mode)mode);
37}
reed@android.com8a1c16f2008-12-17 15:59:43 +000038
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000039void SkAvoidXfermode::flatten(SkWriteBuffer& buffer) const {
djsollen@google.comc73dd5c2012-08-07 15:54:32 +000040 buffer.writeColor(fOpColor);
reed9fa60da2014-08-21 07:59:51 -070041 buffer.writeUInt(fTolerance);
djsollen@google.comc73dd5c2012-08-07 15:54:32 +000042 buffer.writeUInt(fMode);
reed@android.com8a1c16f2008-12-17 15:59:43 +000043}
44
reed@android.com8a1c16f2008-12-17 15:59:43 +000045// returns 0..31
reed@google.com30da7452012-12-17 19:55:24 +000046static unsigned color_dist16(uint16_t c, unsigned r, unsigned g, unsigned b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000047 SkASSERT(r <= SK_R16_MASK);
48 SkASSERT(g <= SK_G16_MASK);
49 SkASSERT(b <= SK_B16_MASK);
50
51 unsigned dr = SkAbs32(SkGetPackedR16(c) - r);
52 unsigned dg = SkAbs32(SkGetPackedG16(c) - g) >> (SK_G16_BITS - SK_R16_BITS);
53 unsigned db = SkAbs32(SkGetPackedB16(c) - b);
tomhudson@google.com1447c6f2011-04-27 14:09:52 +000054
reed@android.com8a1c16f2008-12-17 15:59:43 +000055 return SkMax32(dr, SkMax32(dg, db));
56}
57
reed@android.com8a1c16f2008-12-17 15:59:43 +000058// returns 0..255
reed@google.com30da7452012-12-17 19:55:24 +000059static unsigned color_dist32(SkPMColor c, U8CPU r, U8CPU g, U8CPU b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000060 SkASSERT(r <= 0xFF);
61 SkASSERT(g <= 0xFF);
62 SkASSERT(b <= 0xFF);
63
64 unsigned dr = SkAbs32(SkGetPackedR32(c) - r);
65 unsigned dg = SkAbs32(SkGetPackedG32(c) - g);
66 unsigned db = SkAbs32(SkGetPackedB32(c) - b);
tomhudson@google.com1447c6f2011-04-27 14:09:52 +000067
reed@android.com8a1c16f2008-12-17 15:59:43 +000068 return SkMax32(dr, SkMax32(dg, db));
69}
70
reed@google.com30da7452012-12-17 19:55:24 +000071static int scale_dist_14(int dist, uint32_t mul, uint32_t sub) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000072 int tmp = dist * mul - sub;
73 int result = (tmp + (1 << 13)) >> 14;
74
75 return result;
76}
77
reed@android.com370eeeb2010-03-12 19:23:27 +000078static inline unsigned Accurate255To256(unsigned x) {
79 return x + (x >> 7);
80}
81
reed@android.com8a1c16f2008-12-17 15:59:43 +000082void SkAvoidXfermode::xfer32(SkPMColor dst[], const SkPMColor src[], int count,
reed@google.com30da7452012-12-17 19:55:24 +000083 const SkAlpha aa[]) const {
reed@android.com8a1c16f2008-12-17 15:59:43 +000084 unsigned opR = SkColorGetR(fOpColor);
85 unsigned opG = SkColorGetG(fOpColor);
86 unsigned opB = SkColorGetB(fOpColor);
87 uint32_t mul = fDistMul;
88 uint32_t sub = (fDistMul - (1 << 14)) << 8;
tomhudson@google.com1447c6f2011-04-27 14:09:52 +000089
reed@android.com8a1c16f2008-12-17 15:59:43 +000090 int MAX, mask;
tomhudson@google.com1447c6f2011-04-27 14:09:52 +000091
reed@android.com8a1c16f2008-12-17 15:59:43 +000092 if (kTargetColor_Mode == fMode) {
93 mask = -1;
94 MAX = 255;
95 } else {
96 mask = 0;
97 MAX = 0;
98 }
tomhudson@google.com1447c6f2011-04-27 14:09:52 +000099
reed@android.com8a1c16f2008-12-17 15:59:43 +0000100 for (int i = 0; i < count; i++) {
101 int d = color_dist32(dst[i], opR, opG, opB);
102 // now reverse d if we need to
103 d = MAX + (d ^ mask) - mask;
104 SkASSERT((unsigned)d <= 255);
reed@android.com370eeeb2010-03-12 19:23:27 +0000105 d = Accurate255To256(d);
tomhudson@google.com1447c6f2011-04-27 14:09:52 +0000106
reed@android.com8a1c16f2008-12-17 15:59:43 +0000107 d = scale_dist_14(d, mul, sub);
108 SkASSERT(d <= 256);
tomhudson@google.com1447c6f2011-04-27 14:09:52 +0000109
reed@android.com8a1c16f2008-12-17 15:59:43 +0000110 if (d > 0) {
111 if (NULL != aa) {
reed@android.com370eeeb2010-03-12 19:23:27 +0000112 d = SkAlphaMul(d, Accurate255To256(*aa++));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000113 if (0 == d) {
114 continue;
115 }
116 }
djsollen@google.com8ff1e512012-03-12 14:39:20 +0000117 dst[i] = SkFourByteInterp256(src[i], dst[i], d);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000118 }
119 }
120}
121
reed@google.com30da7452012-12-17 19:55:24 +0000122static inline U16CPU SkBlend3216(SkPMColor src, U16CPU dst, unsigned scale) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000123 SkASSERT(scale <= 32);
124 scale <<= 3;
125
126 return SkPackRGB16( SkAlphaBlend(SkPacked32ToR16(src), SkGetPackedR16(dst), scale),
127 SkAlphaBlend(SkPacked32ToG16(src), SkGetPackedG16(dst), scale),
128 SkAlphaBlend(SkPacked32ToB16(src), SkGetPackedB16(dst), scale));
129}
130
131void SkAvoidXfermode::xfer16(uint16_t dst[], const SkPMColor src[], int count,
reed@google.com30da7452012-12-17 19:55:24 +0000132 const SkAlpha aa[]) const {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000133 unsigned opR = SkColorGetR(fOpColor) >> (8 - SK_R16_BITS);
134 unsigned opG = SkColorGetG(fOpColor) >> (8 - SK_G16_BITS);
135 unsigned opB = SkColorGetB(fOpColor) >> (8 - SK_R16_BITS);
136 uint32_t mul = fDistMul;
reed@android.com0db5a7f2009-11-09 16:01:36 +0000137 uint32_t sub = (fDistMul - (1 << 14)) << SK_R16_BITS;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000138
139 int MAX, mask;
tomhudson@google.com1447c6f2011-04-27 14:09:52 +0000140
reed@android.com8a1c16f2008-12-17 15:59:43 +0000141 if (kTargetColor_Mode == fMode) {
142 mask = -1;
143 MAX = 31;
144 } else {
145 mask = 0;
146 MAX = 0;
147 }
148
149 for (int i = 0; i < count; i++) {
150 int d = color_dist16(dst[i], opR, opG, opB);
151 // now reverse d if we need to
152 d = MAX + (d ^ mask) - mask;
153 SkASSERT((unsigned)d <= 31);
154 // convert from 0..31 to 0..32
155 d += d >> 4;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000156 d = scale_dist_14(d, mul, sub);
157 SkASSERT(d <= 32);
158
159 if (d > 0) {
160 if (NULL != aa) {
reed@android.com370eeeb2010-03-12 19:23:27 +0000161 d = SkAlphaMul(d, Accurate255To256(*aa++));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000162 if (0 == d) {
163 continue;
164 }
165 }
166 dst[i] = SkBlend3216(src[i], dst[i], d);
167 }
168 }
169}
170
reed@google.com30da7452012-12-17 19:55:24 +0000171void SkAvoidXfermode::xferA8(SkAlpha dst[], const SkPMColor src[], int count,
172 const SkAlpha aa[]) const {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000173 // override in subclass
174}
robertphillips@google.comb83b6b42013-01-22 14:32:09 +0000175
commit-bot@chromium.org0f10f7b2014-03-13 18:02:17 +0000176#ifndef SK_IGNORE_TO_STRING
robertphillips@google.comb83b6b42013-01-22 14:32:09 +0000177void SkAvoidXfermode::toString(SkString* str) const {
178 str->append("SkAvoidXfermode: opColor: ");
179 str->appendHex(fOpColor);
180 str->appendf("distMul: %d ", fDistMul);
181
182 static const char* gModeStrings[] = { "Avoid", "Target" };
183
184 str->appendf("mode: %s", gModeStrings[fMode]);
185}
186#endif