blob: a4ae41c689befaf2a746965bad675a673f1d9a6c [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
2/*
3 * Copyright 2006 The Android Open Source Project
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
reed@android.com8a1c16f2008-12-17 15:59:43 +00009
10#include "SkSpriteBlitter.h"
reed@android.comc4cae852009-09-23 15:06:10 +000011#include "SkBlitRow.h"
12#include "SkColorFilter.h"
13#include "SkColorPriv.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000014#include "SkTemplates.h"
15#include "SkUtils.h"
reed@android.comc4cae852009-09-23 15:06:10 +000016#include "SkXfermode.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000017
18///////////////////////////////////////////////////////////////////////////////
19
reed@android.comc4cae852009-09-23 15:06:10 +000020class Sprite_D32_S32 : public SkSpriteBlitter {
reed@android.com8a1c16f2008-12-17 15:59:43 +000021public:
reed@android.comc4cae852009-09-23 15:06:10 +000022 Sprite_D32_S32(const SkBitmap& src, U8CPU alpha) : INHERITED(src) {
commit-bot@chromium.org149e9a12014-04-09 20:45:29 +000023 SkASSERT(src.colorType() == kN32_SkColorType);
reed@android.com8a1c16f2008-12-17 15:59:43 +000024
reed@android.comc4cae852009-09-23 15:06:10 +000025 unsigned flags32 = 0;
26 if (255 != alpha) {
27 flags32 |= SkBlitRow::kGlobalAlpha_Flag32;
28 }
29 if (!src.isOpaque()) {
30 flags32 |= SkBlitRow::kSrcPixelAlpha_Flag32;
31 }
32
33 fProc32 = SkBlitRow::Factory32(flags32);
34 fAlpha = alpha;
35 }
reed@google.com82065d62011-02-07 15:30:46 +000036
reed@android.com8a1c16f2008-12-17 15:59:43 +000037 virtual void blitRect(int x, int y, int width, int height) {
38 SkASSERT(width > 0 && height > 0);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +000039 uint32_t* SK_RESTRICT dst = fDevice->getAddr32(x, y);
40 const uint32_t* SK_RESTRICT src = fSource->getAddr32(x - fLeft,
reed@android.com8a1c16f2008-12-17 15:59:43 +000041 y - fTop);
reed@android.comc4cae852009-09-23 15:06:10 +000042 size_t dstRB = fDevice->rowBytes();
43 size_t srcRB = fSource->rowBytes();
44 SkBlitRow::Proc32 proc = fProc32;
45 U8CPU alpha = fAlpha;
reed@android.com8a1c16f2008-12-17 15:59:43 +000046
47 do {
reed@android.comc4cae852009-09-23 15:06:10 +000048 proc(dst, src, width, alpha);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +000049 dst = (uint32_t* SK_RESTRICT)((char*)dst + dstRB);
50 src = (const uint32_t* SK_RESTRICT)((const char*)src + srcRB);
reed@android.com8a1c16f2008-12-17 15:59:43 +000051 } while (--height != 0);
52 }
reed@android.comc4cae852009-09-23 15:06:10 +000053
54private:
55 SkBlitRow::Proc32 fProc32;
56 U8CPU fAlpha;
reed@google.com82065d62011-02-07 15:30:46 +000057
reed@android.comc4cae852009-09-23 15:06:10 +000058 typedef SkSpriteBlitter INHERITED;
reed@android.com8a1c16f2008-12-17 15:59:43 +000059};
60
61///////////////////////////////////////////////////////////////////////////////
62
reed@android.com8a1c16f2008-12-17 15:59:43 +000063class Sprite_D32_XferFilter : public SkSpriteBlitter {
64public:
65 Sprite_D32_XferFilter(const SkBitmap& source, const SkPaint& paint)
66 : SkSpriteBlitter(source) {
67 fColorFilter = paint.getColorFilter();
reed@google.com82065d62011-02-07 15:30:46 +000068 SkSafeRef(fColorFilter);
69
reed@android.com8a1c16f2008-12-17 15:59:43 +000070 fXfermode = paint.getXfermode();
reed@google.com82065d62011-02-07 15:30:46 +000071 SkSafeRef(fXfermode);
72
reed@android.com8a1c16f2008-12-17 15:59:43 +000073 fBufferSize = 0;
74 fBuffer = NULL;
reed@android.comc4cae852009-09-23 15:06:10 +000075
76 unsigned flags32 = 0;
77 if (255 != paint.getAlpha()) {
78 flags32 |= SkBlitRow::kGlobalAlpha_Flag32;
79 }
80 if (!source.isOpaque()) {
81 flags32 |= SkBlitRow::kSrcPixelAlpha_Flag32;
82 }
reed@google.com82065d62011-02-07 15:30:46 +000083
reed@android.comc4cae852009-09-23 15:06:10 +000084 fProc32 = SkBlitRow::Factory32(flags32);
85 fAlpha = paint.getAlpha();
reed@android.com8a1c16f2008-12-17 15:59:43 +000086 }
reed@google.com82065d62011-02-07 15:30:46 +000087
reed@android.com8a1c16f2008-12-17 15:59:43 +000088 virtual ~Sprite_D32_XferFilter() {
89 delete[] fBuffer;
reed@google.com82065d62011-02-07 15:30:46 +000090 SkSafeUnref(fXfermode);
91 SkSafeUnref(fColorFilter);
reed@android.com8a1c16f2008-12-17 15:59:43 +000092 }
reed@google.com82065d62011-02-07 15:30:46 +000093
reed@android.com8a1c16f2008-12-17 15:59:43 +000094 virtual void setup(const SkBitmap& device, int left, int top,
95 const SkPaint& paint) {
96 this->INHERITED::setup(device, left, top, paint);
reed@google.com82065d62011-02-07 15:30:46 +000097
reed@android.com8a1c16f2008-12-17 15:59:43 +000098 int width = device.width();
99 if (width > fBufferSize) {
100 fBufferSize = width;
101 delete[] fBuffer;
102 fBuffer = new SkPMColor[width];
103 }
104 }
105
106protected:
reed@android.comc4cae852009-09-23 15:06:10 +0000107 SkColorFilter* fColorFilter;
108 SkXfermode* fXfermode;
109 int fBufferSize;
110 SkPMColor* fBuffer;
111 SkBlitRow::Proc32 fProc32;
112 U8CPU fAlpha;
113
reed@android.com8a1c16f2008-12-17 15:59:43 +0000114private:
115 typedef SkSpriteBlitter INHERITED;
116};
117
118///////////////////////////////////////////////////////////////////////////////
119
120class Sprite_D32_S32A_XferFilter : public Sprite_D32_XferFilter {
121public:
122 Sprite_D32_S32A_XferFilter(const SkBitmap& source, const SkPaint& paint)
123 : Sprite_D32_XferFilter(source, paint) {}
124
125 virtual void blitRect(int x, int y, int width, int height) {
126 SkASSERT(width > 0 && height > 0);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000127 uint32_t* SK_RESTRICT dst = fDevice->getAddr32(x, y);
128 const uint32_t* SK_RESTRICT src = fSource->getAddr32(x - fLeft,
reed@android.com8a1c16f2008-12-17 15:59:43 +0000129 y - fTop);
scroggo@google.come5f48242013-02-25 21:47:41 +0000130 size_t dstRB = fDevice->rowBytes();
131 size_t srcRB = fSource->rowBytes();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000132 SkColorFilter* colorFilter = fColorFilter;
133 SkXfermode* xfermode = fXfermode;
134
135 do {
136 const SkPMColor* tmp = src;
reed@google.com82065d62011-02-07 15:30:46 +0000137
reed@android.com8a1c16f2008-12-17 15:59:43 +0000138 if (NULL != colorFilter) {
139 colorFilter->filterSpan(src, width, fBuffer);
140 tmp = fBuffer;
141 }
reed@google.com82065d62011-02-07 15:30:46 +0000142
reed@android.com8a1c16f2008-12-17 15:59:43 +0000143 if (NULL != xfermode) {
144 xfermode->xfer32(dst, tmp, width, NULL);
145 } else {
reed@android.comc4cae852009-09-23 15:06:10 +0000146 fProc32(dst, tmp, width, fAlpha);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000147 }
148
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000149 dst = (uint32_t* SK_RESTRICT)((char*)dst + dstRB);
150 src = (const uint32_t* SK_RESTRICT)((const char*)src + srcRB);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000151 } while (--height != 0);
152 }
reed@google.com82065d62011-02-07 15:30:46 +0000153
reed@android.com8a1c16f2008-12-17 15:59:43 +0000154private:
155 typedef Sprite_D32_XferFilter INHERITED;
156};
157
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000158static void fillbuffer(SkPMColor* SK_RESTRICT dst,
159 const SkPMColor16* SK_RESTRICT src, int count) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000160 SkASSERT(count > 0);
reed@google.com82065d62011-02-07 15:30:46 +0000161
reed@android.com8a1c16f2008-12-17 15:59:43 +0000162 do {
163 *dst++ = SkPixel4444ToPixel32(*src++);
164 } while (--count != 0);
165}
166
167class Sprite_D32_S4444_XferFilter : public Sprite_D32_XferFilter {
168public:
169 Sprite_D32_S4444_XferFilter(const SkBitmap& source, const SkPaint& paint)
170 : Sprite_D32_XferFilter(source, paint) {}
reed@google.com82065d62011-02-07 15:30:46 +0000171
reed@android.com8a1c16f2008-12-17 15:59:43 +0000172 virtual void blitRect(int x, int y, int width, int height) {
173 SkASSERT(width > 0 && height > 0);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000174 SkPMColor* SK_RESTRICT dst = fDevice->getAddr32(x, y);
175 const SkPMColor16* SK_RESTRICT src = fSource->getAddr16(x - fLeft,
reed@android.com8a1c16f2008-12-17 15:59:43 +0000176 y - fTop);
scroggo@google.come5f48242013-02-25 21:47:41 +0000177 size_t dstRB = fDevice->rowBytes();
178 size_t srcRB = fSource->rowBytes();
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000179 SkPMColor* SK_RESTRICT buffer = fBuffer;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000180 SkColorFilter* colorFilter = fColorFilter;
181 SkXfermode* xfermode = fXfermode;
182
183 do {
184 fillbuffer(buffer, src, width);
reed@google.com82065d62011-02-07 15:30:46 +0000185
reed@android.com8a1c16f2008-12-17 15:59:43 +0000186 if (NULL != colorFilter) {
187 colorFilter->filterSpan(buffer, width, buffer);
188 }
189 if (NULL != xfermode) {
190 xfermode->xfer32(dst, buffer, width, NULL);
191 } else {
reed@android.comc4cae852009-09-23 15:06:10 +0000192 fProc32(dst, buffer, width, fAlpha);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000193 }
reed@google.com82065d62011-02-07 15:30:46 +0000194
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000195 dst = (SkPMColor* SK_RESTRICT)((char*)dst + dstRB);
196 src = (const SkPMColor16* SK_RESTRICT)((const char*)src + srcRB);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000197 } while (--height != 0);
198 }
reed@google.com82065d62011-02-07 15:30:46 +0000199
reed@android.com8a1c16f2008-12-17 15:59:43 +0000200private:
201 typedef Sprite_D32_XferFilter INHERITED;
202};
203
204///////////////////////////////////////////////////////////////////////////////
205
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000206static void src_row(SkPMColor* SK_RESTRICT dst,
207 const SkPMColor16* SK_RESTRICT src, int count) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000208 do {
209 *dst = SkPixel4444ToPixel32(*src);
210 src += 1;
211 dst += 1;
212 } while (--count != 0);
213}
214
215class Sprite_D32_S4444_Opaque : public SkSpriteBlitter {
216public:
217 Sprite_D32_S4444_Opaque(const SkBitmap& source) : SkSpriteBlitter(source) {}
reed@google.com82065d62011-02-07 15:30:46 +0000218
reed@android.com8a1c16f2008-12-17 15:59:43 +0000219 virtual void blitRect(int x, int y, int width, int height) {
220 SkASSERT(width > 0 && height > 0);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000221 SkPMColor* SK_RESTRICT dst = fDevice->getAddr32(x, y);
222 const SkPMColor16* SK_RESTRICT src = fSource->getAddr16(x - fLeft,
reed@android.com8a1c16f2008-12-17 15:59:43 +0000223 y - fTop);
scroggo@google.come5f48242013-02-25 21:47:41 +0000224 size_t dstRB = fDevice->rowBytes();
225 size_t srcRB = fSource->rowBytes();
reed@google.com82065d62011-02-07 15:30:46 +0000226
reed@android.com8a1c16f2008-12-17 15:59:43 +0000227 do {
228 src_row(dst, src, width);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000229 dst = (SkPMColor* SK_RESTRICT)((char*)dst + dstRB);
230 src = (const SkPMColor16* SK_RESTRICT)((const char*)src + srcRB);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000231 } while (--height != 0);
232 }
233};
234
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000235static void srcover_row(SkPMColor* SK_RESTRICT dst,
236 const SkPMColor16* SK_RESTRICT src, int count) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000237 do {
238 *dst = SkPMSrcOver(SkPixel4444ToPixel32(*src), *dst);
239 src += 1;
240 dst += 1;
241 } while (--count != 0);
242}
243
244class Sprite_D32_S4444 : public SkSpriteBlitter {
245public:
246 Sprite_D32_S4444(const SkBitmap& source) : SkSpriteBlitter(source) {}
reed@google.com82065d62011-02-07 15:30:46 +0000247
reed@android.com8a1c16f2008-12-17 15:59:43 +0000248 virtual void blitRect(int x, int y, int width, int height) {
249 SkASSERT(width > 0 && height > 0);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000250 SkPMColor* SK_RESTRICT dst = fDevice->getAddr32(x, y);
251 const SkPMColor16* SK_RESTRICT src = fSource->getAddr16(x - fLeft,
reed@android.com8a1c16f2008-12-17 15:59:43 +0000252 y - fTop);
scroggo@google.come5f48242013-02-25 21:47:41 +0000253 size_t dstRB = fDevice->rowBytes();
254 size_t srcRB = fSource->rowBytes();
reed@google.com82065d62011-02-07 15:30:46 +0000255
reed@android.com8a1c16f2008-12-17 15:59:43 +0000256 do {
257 srcover_row(dst, src, width);
tomhudson@google.coma87cd2a2011-06-15 16:50:27 +0000258 dst = (SkPMColor* SK_RESTRICT)((char*)dst + dstRB);
259 src = (const SkPMColor16* SK_RESTRICT)((const char*)src + srcRB);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000260 } while (--height != 0);
261 }
262};
263
264///////////////////////////////////////////////////////////////////////////////
265
commit-bot@chromium.orga5572e52014-03-07 03:24:41 +0000266SkSpriteBlitter* SkSpriteBlitter::ChooseD32(const SkBitmap& source, const SkPaint& paint,
267 SkTBlitterAllocator* allocator) {
268 SkASSERT(allocator != NULL);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000269
reed@android.comc4cae852009-09-23 15:06:10 +0000270 if (paint.getMaskFilter() != NULL) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000271 return NULL;
272 }
273
reed@android.comc4cae852009-09-23 15:06:10 +0000274 U8CPU alpha = paint.getAlpha();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000275 SkXfermode* xfermode = paint.getXfermode();
276 SkColorFilter* filter = paint.getColorFilter();
277 SkSpriteBlitter* blitter = NULL;
278
reed@google.com900ecf22014-02-20 20:55:37 +0000279 switch (source.colorType()) {
280 case kARGB_4444_SkColorType:
reed@android.comc4cae852009-09-23 15:06:10 +0000281 if (alpha != 0xFF) {
282 return NULL; // we only have opaque sprites
283 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000284 if (xfermode || filter) {
commit-bot@chromium.orga5572e52014-03-07 03:24:41 +0000285 blitter = allocator->createT<Sprite_D32_S4444_XferFilter>(source, paint);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000286 } else if (source.isOpaque()) {
commit-bot@chromium.orga5572e52014-03-07 03:24:41 +0000287 blitter = allocator->createT<Sprite_D32_S4444_Opaque>(source);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000288 } else {
commit-bot@chromium.orga5572e52014-03-07 03:24:41 +0000289 blitter = allocator->createT<Sprite_D32_S4444>(source);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000290 }
291 break;
commit-bot@chromium.org149e9a12014-04-09 20:45:29 +0000292 case kN32_SkColorType:
reed@android.com8a1c16f2008-12-17 15:59:43 +0000293 if (xfermode || filter) {
reed@android.comc4cae852009-09-23 15:06:10 +0000294 if (255 == alpha) {
295 // this can handle xfermode or filter, but not alpha
commit-bot@chromium.orga5572e52014-03-07 03:24:41 +0000296 blitter = allocator->createT<Sprite_D32_S32A_XferFilter>(source, paint);
reed@android.comc4cae852009-09-23 15:06:10 +0000297 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000298 } else {
reed@android.comc4cae852009-09-23 15:06:10 +0000299 // this can handle alpha, but not xfermode or filter
commit-bot@chromium.orga5572e52014-03-07 03:24:41 +0000300 blitter = allocator->createT<Sprite_D32_S32>(source, alpha);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000301 }
302 break;
303 default:
304 break;
305 }
306 return blitter;
307}