/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkBlitRow.h" #include "SkBlitMask.h" #include "SkColorPriv.h" #include "SkUtils.h" #define UNROLL SkBlitRow::ColorRectProc PlatformColorRectProcFactory(); static void S32_Opaque_BlitRow32(SkPMColor* SK_RESTRICT dst, const SkPMColor* SK_RESTRICT src, int count, U8CPU alpha) { SkASSERT(255 == alpha); memcpy(dst, src, count * sizeof(SkPMColor)); } static void S32_Blend_BlitRow32(SkPMColor* SK_RESTRICT dst, const SkPMColor* SK_RESTRICT src, int count, U8CPU alpha) { SkASSERT(alpha <= 255); if (count > 0) { unsigned src_scale = SkAlpha255To256(alpha); unsigned dst_scale = 256 - src_scale; #ifdef UNROLL if (count & 1) { *dst = SkAlphaMulQ(*(src++), src_scale) + SkAlphaMulQ(*dst, dst_scale); dst += 1; count -= 1; } const SkPMColor* SK_RESTRICT srcEnd = src + count; while (src != srcEnd) { *dst = SkAlphaMulQ(*(src++), src_scale) + SkAlphaMulQ(*dst, dst_scale); dst += 1; *dst = SkAlphaMulQ(*(src++), src_scale) + SkAlphaMulQ(*dst, dst_scale); dst += 1; } #else do { *dst = SkAlphaMulQ(*src, src_scale) + SkAlphaMulQ(*dst, dst_scale); src += 1; dst += 1; } while (--count > 0); #endif } } static void S32A_Opaque_BlitRow32(SkPMColor* SK_RESTRICT dst, const SkPMColor* SK_RESTRICT src, int count, U8CPU alpha) { SkASSERT(255 == alpha); if (count > 0) { #ifdef UNROLL if (count & 1) { *dst = SkPMSrcOver(*(src++), *dst); dst += 1; count -= 1; } const SkPMColor* SK_RESTRICT srcEnd = src + count; while (src != srcEnd) { *dst = SkPMSrcOver(*(src++), *dst); dst += 1; *dst = SkPMSrcOver(*(src++), *dst); dst += 1; } #else do { *dst = SkPMSrcOver(*src, *dst); src += 1; dst += 1; } while (--count > 0); #endif } } static void S32A_Blend_BlitRow32(SkPMColor* SK_RESTRICT dst, const SkPMColor* SK_RESTRICT src, int count, U8CPU alpha) { SkASSERT(alpha <= 255); if (count > 0) { #ifdef UNROLL if (count & 1) { *dst = SkBlendARGB32(*(src++), *dst, alpha); dst += 1; count -= 1; } const SkPMColor* SK_RESTRICT srcEnd = src + count; while (src != srcEnd) { *dst = SkBlendARGB32(*(src++), *dst, alpha); dst += 1; *dst = SkBlendARGB32(*(src++), *dst, alpha); dst += 1; } #else do { *dst = SkBlendARGB32(*src, *dst, alpha); src += 1; dst += 1; } while (--count > 0); #endif } } /////////////////////////////////////////////////////////////////////////////// static const SkBlitRow::Proc32 gDefault_Procs32[] = { S32_Opaque_BlitRow32, S32_Blend_BlitRow32, S32A_Opaque_BlitRow32, S32A_Blend_BlitRow32 }; SkBlitRow::Proc32 SkBlitRow::Factory32(unsigned flags) { SkASSERT(flags < SK_ARRAY_COUNT(gDefault_Procs32)); // just so we don't crash flags &= kFlags32_Mask; SkBlitRow::Proc32 proc = PlatformProcs32(flags); if (NULL == proc) { proc = gDefault_Procs32[flags]; } SkASSERT(proc); return proc; } SkBlitRow::Proc32 SkBlitRow::ColorProcFactory() { SkBlitRow::ColorProc proc = PlatformColorProc(); if (NULL == proc) { proc = Color32; } SkASSERT(proc); return proc; } void SkBlitRow::Color32(SkPMColor* SK_RESTRICT dst, const SkPMColor* SK_RESTRICT src, int count, SkPMColor color) { if (count > 0) { if (0 == color) { if (src != dst) { memcpy(dst, src, count * sizeof(SkPMColor)); } return; } unsigned colorA = SkGetPackedA32(color); if (255 == colorA) { sk_memset32(dst, color, count); } else { unsigned scale = 256 - SkAlpha255To256(colorA); do { *dst = color + SkAlphaMulQ(*src, scale); src += 1; dst += 1; } while (--count); } } } template void assignLoop(SkPMColor* dst, SkPMColor color) { for (size_t i = 0; i < N; ++i) { *dst++ = color; } } static inline void assignLoop(SkPMColor dst[], SkPMColor color, int count) { while (count >= 4) { *dst++ = color; *dst++ = color; *dst++ = color; *dst++ = color; count -= 4; } if (count >= 2) { *dst++ = color; *dst++ = color; count -= 2; } if (count > 0) { *dst++ = color; } } void SkBlitRow::ColorRect32(SkPMColor* dst, int width, int height, size_t rowBytes, SkPMColor color) { if (width <= 0 || height <= 0 || 0 == color) { return; } // Just made up this value, since I saw it once in a SSE2 file. // We should consider writing some tests to find the optimimal break-point // (or query the Platform proc?) static const int MIN_WIDTH_FOR_SCANLINE_PROC = 32; bool isOpaque = (0xFF == SkGetPackedA32(color)); if (!isOpaque || width >= MIN_WIDTH_FOR_SCANLINE_PROC) { SkBlitRow::ColorProc proc = SkBlitRow::ColorProcFactory(); while (--height >= 0) { (*proc)(dst, dst, width, color); dst = (SkPMColor*) ((char*)dst + rowBytes); } } else { switch (width) { case 1: while (--height >= 0) { assignLoop<1>(dst, color); dst = (SkPMColor*) ((char*)dst + rowBytes); } break; case 2: while (--height >= 0) { assignLoop<2>(dst, color); dst = (SkPMColor*) ((char*)dst + rowBytes); } break; case 3: while (--height >= 0) { assignLoop<3>(dst, color); dst = (SkPMColor*) ((char*)dst + rowBytes); } break; default: while (--height >= 0) { assignLoop(dst, color, width); dst = (SkPMColor*) ((char*)dst + rowBytes); } break; } } } SkBlitRow::ColorRectProc SkBlitRow::ColorRectProcFactory() { SkBlitRow::ColorRectProc proc = PlatformColorRectProcFactory(); if (NULL == proc) { proc = ColorRect32; } SkASSERT(proc); return proc; }