| |
| /* |
| * Copyright 2006 The Android Open Source Project |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| |
| #ifndef SkAntiRun_DEFINED |
| #define SkAntiRun_DEFINED |
| |
| #include "SkBlitter.h" |
| |
| /** Sparse array of run-length-encoded alpha (supersampling coverage) values. |
| Sparseness allows us to independently compose several paths into the |
| same SkAlphaRuns buffer. |
| */ |
| |
| class SkAlphaRuns { |
| public: |
| int16_t* fRuns; |
| uint8_t* fAlpha; |
| |
| /// Returns true if the scanline contains only a single run, |
| /// of alpha value 0. |
| bool empty() const { |
| SkASSERT(fRuns[0] > 0); |
| return fAlpha[0] == 0 && fRuns[fRuns[0]] == 0; |
| } |
| |
| /// Reinitialize for a new scanline. |
| void reset(int width); |
| |
| /** |
| * Insert into the buffer a run starting at (x-offsetX): |
| * if startAlpha > 0 |
| * one pixel with value += startAlpha, |
| * max 255 |
| * if middleCount > 0 |
| * middleCount pixels with value += maxValue |
| * if stopAlpha > 0 |
| * one pixel with value += stopAlpha |
| * Returns the offsetX value that should be passed on the next call, |
| * assuming we're on the same scanline. If the caller is switching |
| * scanlines, then offsetX should be 0 when this is called. |
| */ |
| int add(int x, U8CPU startAlpha, int middleCount, U8CPU stopAlpha, |
| U8CPU maxValue, int offsetX); |
| |
| SkDEBUGCODE(void assertValid(int y, int maxStep) const;) |
| SkDEBUGCODE(void dump() const;) |
| |
| /** |
| * Break the runs in the buffer at offsets x and x+count, properly |
| * updating the runs to the right and left. |
| * i.e. from the state AAAABBBB, run-length encoded as A4B4, |
| * Break(..., 2, 5) would produce AAAABBBB rle as A2A2B3B1. |
| * Allows add() to sum another run to some of the new sub-runs. |
| * i.e. adding ..CCCCC. would produce AADDEEEB, rle as A2D2E3B1. |
| */ |
| static void Break(int16_t runs[], uint8_t alpha[], int x, int count); |
| |
| /** |
| * Cut (at offset x in the buffer) a run into two shorter runs with |
| * matching alpha values. |
| * Used by the RectClipBlitter to trim a RLE encoding to match the |
| * clipping rectangle. |
| */ |
| static void BreakAt(int16_t runs[], uint8_t alpha[], int x) { |
| while (x > 0) { |
| int n = runs[0]; |
| SkASSERT(n > 0); |
| |
| if (x < n) { |
| alpha[x] = alpha[0]; |
| runs[0] = SkToS16(x); |
| runs[x] = SkToS16(n - x); |
| break; |
| } |
| runs += n; |
| alpha += n; |
| x -= n; |
| } |
| } |
| |
| private: |
| SkDEBUGCODE(int fWidth;) |
| SkDEBUGCODE(void validate() const;) |
| }; |
| |
| #endif |