• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2011 Google Inc.
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 
8 #include "SkGpuDevice.h"
9 
10 #include "effects/GrTextureDomainEffect.h"
11 #include "effects/GrSimpleTextureEffect.h"
12 
13 #include "GrContext.h"
14 #include "GrTextContext.h"
15 
16 #include "SkGrTexturePixelRef.h"
17 
18 #include "SkColorFilter.h"
19 #include "SkDeviceImageFilterProxy.h"
20 #include "SkDrawProcs.h"
21 #include "SkGlyphCache.h"
22 #include "SkImageFilter.h"
23 #include "SkPathEffect.h"
24 #include "SkStroke.h"
25 #include "SkUtils.h"
26 
27 #define CACHE_COMPATIBLE_DEVICE_TEXTURES 1
28 
29 #if 0
30     extern bool (*gShouldDrawProc)();
31     #define CHECK_SHOULD_DRAW(draw, forceI)                     \
32         do {                                                    \
33             if (gShouldDrawProc && !gShouldDrawProc()) return;  \
34             this->prepareDraw(draw, forceI);                    \
35         } while (0)
36 #else
37     #define CHECK_SHOULD_DRAW(draw, forceI) this->prepareDraw(draw, forceI)
38 #endif
39 
40 // we use the same texture slot on GrPaint for bitmaps and shaders
41 // (since drawBitmap, drawSprite, and drawDevice ignore skia's shader)
42 enum {
43     kBitmapTextureIdx = 0,
44     kShaderTextureIdx = 0,
45     kColorFilterTextureIdx = 1
46 };
47 
48 #define MAX_BLUR_SIGMA 4.0f
49 // FIXME:  This value comes from from SkBlurMaskFilter.cpp.
50 // Should probably be put in a common header someplace.
51 #define MAX_BLUR_RADIUS SkIntToScalar(128)
52 // This constant approximates the scaling done in the software path's
53 // "high quality" mode, in SkBlurMask::Blur() (1 / sqrt(3)).
54 // IMHO, it actually should be 1:  we blur "less" than we should do
55 // according to the CSS and canvas specs, simply because Safari does the same.
56 // Firefox used to do the same too, until 4.0 where they fixed it.  So at some
57 // point we should probably get rid of these scaling constants and rebaseline
58 // all the blur tests.
59 #define BLUR_SIGMA_SCALE 0.6f
60 // This constant represents the screen alignment criterion in texels for
61 // requiring texture domain clamping to prevent color bleeding when drawing
62 // a sub region of a larger source image.
63 #define COLOR_BLEED_TOLERANCE SkFloatToScalar(0.001f)
64 
65 #define DO_DEFERRED_CLEAR()             \
66     do {                                \
67         if (fNeedClear) {               \
68             this->clear(SK_ColorTRANSPARENT); \
69         }                               \
70     } while (false)                     \
71 
72 ///////////////////////////////////////////////////////////////////////////////
73 
74 #define CHECK_FOR_NODRAW_ANNOTATION(paint) \
75     do { if (paint.isNoDrawAnnotation()) { return; } } while (0)
76 
77 ///////////////////////////////////////////////////////////////////////////////
78 
79 
80 class SkGpuDevice::SkAutoCachedTexture : public ::SkNoncopyable {
81 public:
SkAutoCachedTexture()82     SkAutoCachedTexture()
83         : fDevice(NULL)
84         , fTexture(NULL) {
85     }
86 
SkAutoCachedTexture(SkGpuDevice * device,const SkBitmap & bitmap,const GrTextureParams * params,GrTexture ** texture)87     SkAutoCachedTexture(SkGpuDevice* device,
88                         const SkBitmap& bitmap,
89                         const GrTextureParams* params,
90                         GrTexture** texture)
91         : fDevice(NULL)
92         , fTexture(NULL) {
93         GrAssert(NULL != texture);
94         *texture = this->set(device, bitmap, params);
95     }
96 
~SkAutoCachedTexture()97     ~SkAutoCachedTexture() {
98         if (NULL != fTexture) {
99             GrUnlockAndUnrefCachedBitmapTexture(fTexture);
100         }
101     }
102 
set(SkGpuDevice * device,const SkBitmap & bitmap,const GrTextureParams * params)103     GrTexture* set(SkGpuDevice* device,
104                    const SkBitmap& bitmap,
105                    const GrTextureParams* params) {
106         if (NULL != fTexture) {
107             GrUnlockAndUnrefCachedBitmapTexture(fTexture);
108             fTexture = NULL;
109         }
110         fDevice = device;
111         GrTexture* result = (GrTexture*)bitmap.getTexture();
112         if (NULL == result) {
113             // Cannot return the native texture so look it up in our cache
114             fTexture = GrLockAndRefCachedBitmapTexture(device->context(), bitmap, params);
115             result = fTexture;
116         }
117         return result;
118     }
119 
120 private:
121     SkGpuDevice* fDevice;
122     GrTexture*   fTexture;
123 };
124 
125 ///////////////////////////////////////////////////////////////////////////////
126 
127 struct GrSkDrawProcs : public SkDrawProcs {
128 public:
129     GrContext* fContext;
130     GrTextContext* fTextContext;
131     GrFontScaler* fFontScaler;  // cached in the skia glyphcache
132 };
133 
134 ///////////////////////////////////////////////////////////////////////////////
135 
grConfig2skConfig(GrPixelConfig config,bool * isOpaque)136 static SkBitmap::Config grConfig2skConfig(GrPixelConfig config, bool* isOpaque) {
137     switch (config) {
138         case kAlpha_8_GrPixelConfig:
139             *isOpaque = false;
140             return SkBitmap::kA8_Config;
141         case kRGB_565_GrPixelConfig:
142             *isOpaque = true;
143             return SkBitmap::kRGB_565_Config;
144         case kRGBA_4444_GrPixelConfig:
145             *isOpaque = false;
146             return SkBitmap::kARGB_4444_Config;
147         case kSkia8888_PM_GrPixelConfig:
148             // we don't currently have a way of knowing whether
149             // a 8888 is opaque based on the config.
150             *isOpaque = false;
151             return SkBitmap::kARGB_8888_Config;
152         default:
153             *isOpaque = false;
154             return SkBitmap::kNo_Config;
155     }
156 }
157 
make_bitmap(GrContext * context,GrRenderTarget * renderTarget)158 static SkBitmap make_bitmap(GrContext* context, GrRenderTarget* renderTarget) {
159     GrPixelConfig config = renderTarget->config();
160 
161     bool isOpaque;
162     SkBitmap bitmap;
163     bitmap.setConfig(grConfig2skConfig(config, &isOpaque),
164                      renderTarget->width(), renderTarget->height());
165     bitmap.setIsOpaque(isOpaque);
166     return bitmap;
167 }
168 
SkGpuDevice(GrContext * context,GrTexture * texture)169 SkGpuDevice::SkGpuDevice(GrContext* context, GrTexture* texture)
170 : SkDevice(make_bitmap(context, texture->asRenderTarget())) {
171     this->initFromRenderTarget(context, texture->asRenderTarget(), false);
172 }
173 
SkGpuDevice(GrContext * context,GrRenderTarget * renderTarget)174 SkGpuDevice::SkGpuDevice(GrContext* context, GrRenderTarget* renderTarget)
175 : SkDevice(make_bitmap(context, renderTarget)) {
176     this->initFromRenderTarget(context, renderTarget, false);
177 }
178 
initFromRenderTarget(GrContext * context,GrRenderTarget * renderTarget,bool cached)179 void SkGpuDevice::initFromRenderTarget(GrContext* context,
180                                        GrRenderTarget* renderTarget,
181                                        bool cached) {
182     fDrawProcs = NULL;
183 
184     fContext = context;
185     fContext->ref();
186 
187     fRenderTarget = NULL;
188     fNeedClear = false;
189 
190     GrAssert(NULL != renderTarget);
191     fRenderTarget = renderTarget;
192     fRenderTarget->ref();
193 
194     // Hold onto to the texture in the pixel ref (if there is one) because the texture holds a ref
195     // on the RT but not vice-versa.
196     // TODO: Remove this trickery once we figure out how to make SkGrPixelRef do this without
197     // busting chrome (for a currently unknown reason).
198     GrSurface* surface = fRenderTarget->asTexture();
199     if (NULL == surface) {
200         surface = fRenderTarget;
201     }
202     SkPixelRef* pr = SkNEW_ARGS(SkGrPixelRef, (surface, cached));
203 
204     this->setPixelRef(pr, 0)->unref();
205 }
206 
SkGpuDevice(GrContext * context,SkBitmap::Config config,int width,int height,int sampleCount)207 SkGpuDevice::SkGpuDevice(GrContext* context,
208                          SkBitmap::Config config,
209                          int width,
210                          int height,
211                          int sampleCount)
212     : SkDevice(config, width, height, false /*isOpaque*/) {
213 
214     fDrawProcs = NULL;
215 
216     fContext = context;
217     fContext->ref();
218 
219     fRenderTarget = NULL;
220     fNeedClear = false;
221 
222     if (config != SkBitmap::kRGB_565_Config) {
223         config = SkBitmap::kARGB_8888_Config;
224     }
225 
226     GrTextureDesc desc;
227     desc.fFlags = kRenderTarget_GrTextureFlagBit;
228     desc.fWidth = width;
229     desc.fHeight = height;
230     desc.fConfig = SkBitmapConfig2GrPixelConfig(config);
231     desc.fSampleCnt = sampleCount;
232 
233     SkAutoTUnref<GrTexture> texture(fContext->createUncachedTexture(desc, NULL, 0));
234 
235     if (NULL != texture) {
236         fRenderTarget = texture->asRenderTarget();
237         fRenderTarget->ref();
238 
239         GrAssert(NULL != fRenderTarget);
240 
241         // wrap the bitmap with a pixelref to expose our texture
242         SkGrPixelRef* pr = SkNEW_ARGS(SkGrPixelRef, (texture));
243         this->setPixelRef(pr, 0)->unref();
244     } else {
245         GrPrintf("--- failed to create gpu-offscreen [%d %d]\n",
246                  width, height);
247         GrAssert(false);
248     }
249 }
250 
~SkGpuDevice()251 SkGpuDevice::~SkGpuDevice() {
252     if (fDrawProcs) {
253         delete fDrawProcs;
254     }
255 
256     // The GrContext takes a ref on the target. We don't want to cause the render
257     // target to be unnecessarily kept alive.
258     if (fContext->getRenderTarget() == fRenderTarget) {
259         fContext->setRenderTarget(NULL);
260     }
261 
262     if (fContext->getClip() == &fClipData) {
263         fContext->setClip(NULL);
264     }
265 
266     SkSafeUnref(fRenderTarget);
267     fContext->unref();
268 }
269 
270 ///////////////////////////////////////////////////////////////////////////////
271 
makeRenderTargetCurrent()272 void SkGpuDevice::makeRenderTargetCurrent() {
273     DO_DEFERRED_CLEAR();
274     fContext->setRenderTarget(fRenderTarget);
275 }
276 
277 ///////////////////////////////////////////////////////////////////////////////
278 
279 namespace {
config8888_to_grconfig_and_flags(SkCanvas::Config8888 config8888,uint32_t * flags)280 GrPixelConfig config8888_to_grconfig_and_flags(SkCanvas::Config8888 config8888, uint32_t* flags) {
281     switch (config8888) {
282         case SkCanvas::kNative_Premul_Config8888:
283             *flags = 0;
284             return kSkia8888_GrPixelConfig;
285         case SkCanvas::kNative_Unpremul_Config8888:
286             *flags = GrContext::kUnpremul_PixelOpsFlag;
287             return kSkia8888_PM_GrPixelConfig;
288         case SkCanvas::kBGRA_Premul_Config8888:
289             *flags = 0;
290             return kBGRA_8888_GrPixelConfig;
291         case SkCanvas::kBGRA_Unpremul_Config8888:
292             *flags = GrContext::kUnpremul_PixelOpsFlag;
293             return kBGRA_8888_GrPixelConfig;
294         case SkCanvas::kRGBA_Premul_Config8888:
295             *flags = 0;
296             return kRGBA_8888_GrPixelConfig;
297         case SkCanvas::kRGBA_Unpremul_Config8888:
298             *flags = GrContext::kUnpremul_PixelOpsFlag;
299             return kRGBA_8888_GrPixelConfig;
300         default:
301             GrCrash("Unexpected Config8888.");
302             *flags = 0; // suppress warning
303             return kSkia8888_PM_GrPixelConfig;
304     }
305 }
306 }
307 
onReadPixels(const SkBitmap & bitmap,int x,int y,SkCanvas::Config8888 config8888)308 bool SkGpuDevice::onReadPixels(const SkBitmap& bitmap,
309                                int x, int y,
310                                SkCanvas::Config8888 config8888) {
311     DO_DEFERRED_CLEAR();
312     SkASSERT(SkBitmap::kARGB_8888_Config == bitmap.config());
313     SkASSERT(!bitmap.isNull());
314     SkASSERT(SkIRect::MakeWH(this->width(), this->height()).contains(SkIRect::MakeXYWH(x, y, bitmap.width(), bitmap.height())));
315 
316     SkAutoLockPixels alp(bitmap);
317     GrPixelConfig config;
318     uint32_t flags;
319     config = config8888_to_grconfig_and_flags(config8888, &flags);
320     return fContext->readRenderTargetPixels(fRenderTarget,
321                                             x, y,
322                                             bitmap.width(),
323                                             bitmap.height(),
324                                             config,
325                                             bitmap.getPixels(),
326                                             bitmap.rowBytes(),
327                                             flags);
328 }
329 
writePixels(const SkBitmap & bitmap,int x,int y,SkCanvas::Config8888 config8888)330 void SkGpuDevice::writePixels(const SkBitmap& bitmap, int x, int y,
331                               SkCanvas::Config8888 config8888) {
332     SkAutoLockPixels alp(bitmap);
333     if (!bitmap.readyToDraw()) {
334         return;
335     }
336 
337     GrPixelConfig config;
338     uint32_t flags;
339     if (SkBitmap::kARGB_8888_Config == bitmap.config()) {
340         config = config8888_to_grconfig_and_flags(config8888, &flags);
341     } else {
342         flags = 0;
343         config= SkBitmapConfig2GrPixelConfig(bitmap.config());
344     }
345 
346     fRenderTarget->writePixels(x, y, bitmap.width(), bitmap.height(),
347                                config, bitmap.getPixels(), bitmap.rowBytes(), flags);
348 }
349 
350 namespace {
purgeClipCB(int genID,void *)351 void purgeClipCB(int genID, void* ) {
352 
353     if (SkClipStack::kInvalidGenID == genID ||
354         SkClipStack::kEmptyGenID == genID ||
355         SkClipStack::kWideOpenGenID == genID) {
356         // none of these cases will have a cached clip mask
357         return;
358     }
359 
360 }
361 };
362 
onAttachToCanvas(SkCanvas * canvas)363 void SkGpuDevice::onAttachToCanvas(SkCanvas* canvas) {
364     INHERITED::onAttachToCanvas(canvas);
365 
366     // Canvas promises that this ptr is valid until onDetachFromCanvas is called
367     fClipData.fClipStack = canvas->getClipStack();
368 
369     fClipData.fClipStack->addPurgeClipCallback(purgeClipCB, fContext);
370 }
371 
onDetachFromCanvas()372 void SkGpuDevice::onDetachFromCanvas() {
373     INHERITED::onDetachFromCanvas();
374 
375     // TODO: iterate through the clip stack and clean up any cached clip masks
376     fClipData.fClipStack->removePurgeClipCallback(purgeClipCB, fContext);
377 
378     fClipData.fClipStack = NULL;
379 }
380 
381 #ifdef SK_DEBUG
check_bounds(const GrClipData & clipData,const SkRegion & clipRegion,int renderTargetWidth,int renderTargetHeight)382 static void check_bounds(const GrClipData& clipData,
383                          const SkRegion& clipRegion,
384                          int renderTargetWidth,
385                          int renderTargetHeight) {
386 
387     SkIRect devBound;
388 
389     devBound.setLTRB(0, 0, renderTargetWidth, renderTargetHeight);
390 
391     SkClipStack::BoundsType boundType;
392     SkRect canvTemp;
393 
394     clipData.fClipStack->getBounds(&canvTemp, &boundType);
395     if (SkClipStack::kNormal_BoundsType == boundType) {
396         SkIRect devTemp;
397 
398         canvTemp.roundOut(&devTemp);
399 
400         devTemp.offset(-clipData.fOrigin.fX, -clipData.fOrigin.fY);
401 
402         if (!devBound.intersect(devTemp)) {
403             devBound.setEmpty();
404         }
405     }
406 
407     GrAssert(devBound.contains(clipRegion.getBounds()));
408 }
409 #endif
410 
411 ///////////////////////////////////////////////////////////////////////////////
412 
413 // call this every draw call, to ensure that the context reflects our state,
414 // and not the state from some other canvas/device
prepareDraw(const SkDraw & draw,bool forceIdentity)415 void SkGpuDevice::prepareDraw(const SkDraw& draw, bool forceIdentity) {
416     GrAssert(NULL != fClipData.fClipStack);
417 
418     fContext->setRenderTarget(fRenderTarget);
419 
420     SkASSERT(draw.fClipStack && draw.fClipStack == fClipData.fClipStack);
421 
422     if (forceIdentity) {
423         fContext->setIdentityMatrix();
424     } else {
425         fContext->setMatrix(*draw.fMatrix);
426     }
427     fClipData.fOrigin = this->getOrigin();
428 
429 #ifdef SK_DEBUG
430     check_bounds(fClipData, *draw.fClip, fRenderTarget->width(), fRenderTarget->height());
431 #endif
432 
433     fContext->setClip(&fClipData);
434 
435     DO_DEFERRED_CLEAR();
436 }
437 
accessRenderTarget()438 SkGpuRenderTarget* SkGpuDevice::accessRenderTarget() {
439     DO_DEFERRED_CLEAR();
440     return (SkGpuRenderTarget*)fRenderTarget;
441 }
442 
bindDeviceAsTexture(GrPaint * paint)443 bool SkGpuDevice::bindDeviceAsTexture(GrPaint* paint) {
444     GrTexture* texture = fRenderTarget->asTexture();
445     if (NULL != texture) {
446         paint->colorStage(kBitmapTextureIdx)->setEffect(
447             GrSimpleTextureEffect::Create(texture, SkMatrix::I()))->unref();
448         return true;
449     }
450     return false;
451 }
452 
453 ///////////////////////////////////////////////////////////////////////////////
454 
455 SK_COMPILE_ASSERT(SkShader::kNone_BitmapType == 0, shader_type_mismatch);
456 SK_COMPILE_ASSERT(SkShader::kDefault_BitmapType == 1, shader_type_mismatch);
457 SK_COMPILE_ASSERT(SkShader::kRadial_BitmapType == 2, shader_type_mismatch);
458 SK_COMPILE_ASSERT(SkShader::kSweep_BitmapType == 3, shader_type_mismatch);
459 SK_COMPILE_ASSERT(SkShader::kTwoPointRadial_BitmapType == 4,
460                   shader_type_mismatch);
461 SK_COMPILE_ASSERT(SkShader::kTwoPointConical_BitmapType == 5,
462                   shader_type_mismatch);
463 SK_COMPILE_ASSERT(SkShader::kLinear_BitmapType == 6, shader_type_mismatch);
464 SK_COMPILE_ASSERT(SkShader::kLast_BitmapType == 6, shader_type_mismatch);
465 
466 namespace {
467 
468 // converts a SkPaint to a GrPaint, ignoring the skPaint's shader
469 // justAlpha indicates that skPaint's alpha should be used rather than the color
470 // Callers may subsequently modify the GrPaint. Setting constantColor indicates
471 // that the final paint will draw the same color at every pixel. This allows
472 // an optimization where the the color filter can be applied to the skPaint's
473 // color once while converting to GrPaint and then ignored.
skPaint2GrPaintNoShader(SkGpuDevice * dev,const SkPaint & skPaint,bool justAlpha,bool constantColor,GrPaint * grPaint)474 inline bool skPaint2GrPaintNoShader(SkGpuDevice* dev,
475                                     const SkPaint& skPaint,
476                                     bool justAlpha,
477                                     bool constantColor,
478                                     GrPaint* grPaint) {
479 
480     grPaint->setDither(skPaint.isDither());
481     grPaint->setAntiAlias(skPaint.isAntiAlias());
482 
483     SkXfermode::Coeff sm = SkXfermode::kOne_Coeff;
484     SkXfermode::Coeff dm = SkXfermode::kISA_Coeff;
485 
486     SkXfermode* mode = skPaint.getXfermode();
487     if (mode) {
488         if (!mode->asCoeff(&sm, &dm)) {
489             //SkDEBUGCODE(SkDebugf("Unsupported xfer mode.\n");)
490 #if 0
491             return false;
492 #endif
493         }
494     }
495     grPaint->setBlendFunc(sk_blend_to_grblend(sm), sk_blend_to_grblend(dm));
496 
497     if (justAlpha) {
498         uint8_t alpha = skPaint.getAlpha();
499         grPaint->setColor(GrColorPackRGBA(alpha, alpha, alpha, alpha));
500         // justAlpha is currently set to true only if there is a texture,
501         // so constantColor should not also be true.
502         GrAssert(!constantColor);
503     } else {
504         grPaint->setColor(SkColor2GrColor(skPaint.getColor()));
505         GrAssert(!grPaint->isColorStageEnabled(kShaderTextureIdx));
506     }
507 
508     SkColorFilter* colorFilter = skPaint.getColorFilter();
509     if (NULL != colorFilter) {
510         // if the source color is a constant then apply the filter here once rather than per pixel
511         // in a shader.
512         if (constantColor) {
513             SkColor filtered = colorFilter->filterColor(skPaint.getColor());
514             grPaint->setColor(SkColor2GrColor(filtered));
515         } else {
516             SkAutoTUnref<GrEffectRef> effect(colorFilter->asNewEffect(dev->context()));
517             if (NULL != effect.get()) {
518                 grPaint->colorStage(kColorFilterTextureIdx)->setEffect(effect);
519             } else {
520                 // TODO: rewrite this using asNewEffect()
521                 SkColor color;
522                 SkXfermode::Mode filterMode;
523                 if (colorFilter->asColorMode(&color, &filterMode)) {
524                     grPaint->setXfermodeColorFilter(filterMode, SkColor2GrColor(color));
525                 }
526             }
527         }
528     }
529 
530     return true;
531 }
532 
533 // This function is similar to skPaint2GrPaintNoShader but also converts
534 // skPaint's shader to a GrTexture/GrEffectStage if possible. The texture to
535 // be used is set on grPaint and returned in param act. constantColor has the
536 // same meaning as in skPaint2GrPaintNoShader.
skPaint2GrPaintShader(SkGpuDevice * dev,const SkPaint & skPaint,bool constantColor,GrPaint * grPaint)537 inline bool skPaint2GrPaintShader(SkGpuDevice* dev,
538                                   const SkPaint& skPaint,
539                                   bool constantColor,
540                                   GrPaint* grPaint) {
541     SkShader* shader = skPaint.getShader();
542     if (NULL == shader) {
543         return skPaint2GrPaintNoShader(dev, skPaint, false, constantColor, grPaint);
544     } else if (!skPaint2GrPaintNoShader(dev, skPaint, true, false, grPaint)) {
545         return false;
546     }
547 
548     SkAutoTUnref<GrEffectRef> effect(shader->asNewEffect(dev->context(), skPaint));
549     if (NULL != effect.get()) {
550         grPaint->colorStage(kShaderTextureIdx)->setEffect(effect);
551         return true;
552     }
553 
554     // We still don't have SkColorShader::asNewEffect() implemented.
555     SkShader::GradientInfo info;
556     SkColor                color;
557 
558     info.fColors = &color;
559     info.fColorOffsets = NULL;
560     info.fColorCount = 1;
561     if (SkShader::kColor_GradientType == shader->asAGradient(&info)) {
562         SkPaint copy(skPaint);
563         copy.setShader(NULL);
564         // modulate the paint alpha by the shader's solid color alpha
565         U8CPU newA = SkMulDiv255Round(SkColorGetA(color), copy.getAlpha());
566         copy.setColor(SkColorSetA(color, newA));
567         return skPaint2GrPaintNoShader(dev, copy, false, constantColor, grPaint);
568     }
569     return false;
570 }
571 }
572 
573 ///////////////////////////////////////////////////////////////////////////////
clear(SkColor color)574 void SkGpuDevice::clear(SkColor color) {
575     SkIRect rect = SkIRect::MakeWH(this->width(), this->height());
576     fContext->clear(&rect, SkColor2GrColor(color), fRenderTarget);
577     fNeedClear = false;
578 }
579 
drawPaint(const SkDraw & draw,const SkPaint & paint)580 void SkGpuDevice::drawPaint(const SkDraw& draw, const SkPaint& paint) {
581     CHECK_SHOULD_DRAW(draw, false);
582 
583     GrPaint grPaint;
584     if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
585         return;
586     }
587 
588     fContext->drawPaint(grPaint);
589 }
590 
591 // must be in SkCanvas::PointMode order
592 static const GrPrimitiveType gPointMode2PrimtiveType[] = {
593     kPoints_GrPrimitiveType,
594     kLines_GrPrimitiveType,
595     kLineStrip_GrPrimitiveType
596 };
597 
drawPoints(const SkDraw & draw,SkCanvas::PointMode mode,size_t count,const SkPoint pts[],const SkPaint & paint)598 void SkGpuDevice::drawPoints(const SkDraw& draw, SkCanvas::PointMode mode,
599                              size_t count, const SkPoint pts[], const SkPaint& paint) {
600     CHECK_SHOULD_DRAW(draw, false);
601 
602     SkScalar width = paint.getStrokeWidth();
603     if (width < 0) {
604         return;
605     }
606 
607     // we only handle hairlines and paints without path effects or mask filters,
608     // else we let the SkDraw call our drawPath()
609     if (width > 0 || paint.getPathEffect() || paint.getMaskFilter()) {
610         draw.drawPoints(mode, count, pts, paint, true);
611         return;
612     }
613 
614     GrPaint grPaint;
615     if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
616         return;
617     }
618 
619     fContext->drawVertices(grPaint,
620                            gPointMode2PrimtiveType[mode],
621                            count,
622                            (GrPoint*)pts,
623                            NULL,
624                            NULL,
625                            NULL,
626                            0);
627 }
628 
629 ///////////////////////////////////////////////////////////////////////////////
630 
drawRect(const SkDraw & draw,const SkRect & rect,const SkPaint & paint)631 void SkGpuDevice::drawRect(const SkDraw& draw, const SkRect& rect,
632                            const SkPaint& paint) {
633     CHECK_FOR_NODRAW_ANNOTATION(paint);
634     CHECK_SHOULD_DRAW(draw, false);
635 
636     bool doStroke = paint.getStyle() != SkPaint::kFill_Style;
637     SkScalar width = paint.getStrokeWidth();
638 
639     /*
640         We have special code for hairline strokes, miter-strokes, and fills.
641         Anything else we just call our path code.
642      */
643     bool usePath = doStroke && width > 0 &&
644                     paint.getStrokeJoin() != SkPaint::kMiter_Join;
645     // another two reasons we might need to call drawPath...
646     if (paint.getMaskFilter() || paint.getPathEffect()) {
647         usePath = true;
648     }
649     // until we aa rotated rects...
650     if (!usePath && paint.isAntiAlias() && !fContext->getMatrix().rectStaysRect()) {
651         usePath = true;
652     }
653     // small miter limit means right angles show bevel...
654     if (SkPaint::kMiter_Join == paint.getStrokeJoin() &&
655         paint.getStrokeMiter() < SK_ScalarSqrt2)
656     {
657         usePath = true;
658     }
659     // until we can both stroke and fill rectangles
660     if (paint.getStyle() == SkPaint::kStrokeAndFill_Style) {
661         usePath = true;
662     }
663 
664     if (usePath) {
665         SkPath path;
666         path.addRect(rect);
667         this->drawPath(draw, path, paint, NULL, true);
668         return;
669     }
670 
671     GrPaint grPaint;
672     if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
673         return;
674     }
675     fContext->drawRect(grPaint, rect, doStroke ? width : -1);
676 }
677 
678 ///////////////////////////////////////////////////////////////////////////////
679 
drawOval(const SkDraw & draw,const SkRect & oval,const SkPaint & paint)680 void SkGpuDevice::drawOval(const SkDraw& draw, const SkRect& oval,
681                            const SkPaint& paint) {
682     CHECK_FOR_NODRAW_ANNOTATION(paint);
683     CHECK_SHOULD_DRAW(draw, false);
684 
685     bool usePath = false;
686     // some basic reasons we might need to call drawPath...
687     if (paint.getMaskFilter() || paint.getPathEffect()) {
688         usePath = true;
689     }
690 
691     if (usePath) {
692         SkPath path;
693         path.addOval(oval);
694         this->drawPath(draw, path, paint, NULL, true);
695         return;
696     }
697 
698     GrPaint grPaint;
699     if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
700         return;
701     }
702     SkStrokeRec stroke(paint);
703 
704     fContext->drawOval(grPaint, oval, stroke);
705 }
706 
707 #include "SkMaskFilter.h"
708 #include "SkBounder.h"
709 
710 ///////////////////////////////////////////////////////////////////////////////
711 
712 // helpers for applying mask filters
713 namespace {
714 
715 // We prefer to blur small rect with small radius via CPU.
716 #define MIN_GPU_BLUR_SIZE SkIntToScalar(64)
717 #define MIN_GPU_BLUR_RADIUS SkIntToScalar(32)
shouldDrawBlurWithCPU(const SkRect & rect,SkScalar radius)718 inline bool shouldDrawBlurWithCPU(const SkRect& rect, SkScalar radius) {
719     if (rect.width() <= MIN_GPU_BLUR_SIZE &&
720         rect.height() <= MIN_GPU_BLUR_SIZE &&
721         radius <= MIN_GPU_BLUR_RADIUS) {
722         return true;
723     }
724     return false;
725 }
726 
drawWithGPUMaskFilter(GrContext * context,const SkPath & devPath,const SkStrokeRec & stroke,SkMaskFilter * filter,const SkRegion & clip,SkBounder * bounder,GrPaint * grp)727 bool drawWithGPUMaskFilter(GrContext* context, const SkPath& devPath, const SkStrokeRec& stroke,
728                            SkMaskFilter* filter, const SkRegion& clip,
729                            SkBounder* bounder, GrPaint* grp) {
730     SkMaskFilter::BlurInfo info;
731     SkMaskFilter::BlurType blurType = filter->asABlur(&info);
732     if (SkMaskFilter::kNone_BlurType == blurType) {
733         return false;
734     }
735     SkScalar radius = info.fIgnoreTransform ? info.fRadius
736                                             : context->getMatrix().mapRadius(info.fRadius);
737     radius = SkMinScalar(radius, MAX_BLUR_RADIUS);
738     if (radius <= 0) {
739         return false;
740     }
741 
742     SkRect srcRect = devPath.getBounds();
743     if (shouldDrawBlurWithCPU(srcRect, radius)) {
744         return false;
745     }
746 
747     float sigma = SkScalarToFloat(radius) * BLUR_SIGMA_SCALE;
748     float sigma3 = sigma * 3.0f;
749 
750     SkRect clipRect;
751     clipRect.set(clip.getBounds());
752 
753     // Outset srcRect and clipRect by 3 * sigma, to compute affected blur area.
754     srcRect.inset(SkFloatToScalar(-sigma3), SkFloatToScalar(-sigma3));
755     clipRect.inset(SkFloatToScalar(-sigma3), SkFloatToScalar(-sigma3));
756     srcRect.intersect(clipRect);
757     SkRect finalRect = srcRect;
758     SkIRect finalIRect;
759     finalRect.roundOut(&finalIRect);
760     if (clip.quickReject(finalIRect)) {
761         return true;
762     }
763     if (bounder && !bounder->doIRect(finalIRect)) {
764         return true;
765     }
766     GrPoint offset = GrPoint::Make(-srcRect.fLeft, -srcRect.fTop);
767     srcRect.offset(offset);
768     GrTextureDesc desc;
769     desc.fFlags = kRenderTarget_GrTextureFlagBit;
770     desc.fWidth = SkScalarCeilToInt(srcRect.width());
771     desc.fHeight = SkScalarCeilToInt(srcRect.height());
772     // We actually only need A8, but it often isn't supported as a
773     // render target so default to RGBA_8888
774     desc.fConfig = kRGBA_8888_GrPixelConfig;
775 
776     if (context->isConfigRenderable(kAlpha_8_GrPixelConfig)) {
777         desc.fConfig = kAlpha_8_GrPixelConfig;
778     }
779 
780     GrAutoScratchTexture pathEntry(context, desc);
781     GrTexture* pathTexture = pathEntry.texture();
782     if (NULL == pathTexture) {
783         return false;
784     }
785 
786     SkAutoTUnref<GrTexture> blurTexture;
787 
788     {
789         GrContext::AutoRenderTarget art(context, pathTexture->asRenderTarget());
790         GrContext::AutoClip ac(context, srcRect);
791 
792         context->clear(NULL, 0);
793 
794         GrPaint tempPaint;
795         if (grp->isAntiAlias()) {
796             tempPaint.setAntiAlias(true);
797             // AA uses the "coverage" stages on GrDrawTarget. Coverage with a dst
798             // blend coeff of zero requires dual source blending support in order
799             // to properly blend partially covered pixels. This means the AA
800             // code path may not be taken. So we use a dst blend coeff of ISA. We
801             // could special case AA draws to a dst surface with known alpha=0 to
802             // use a zero dst coeff when dual source blending isn't available.f
803             tempPaint.setBlendFunc(kOne_GrBlendCoeff, kISC_GrBlendCoeff);
804         }
805 
806         GrContext::AutoMatrix am;
807 
808         // Draw hard shadow to pathTexture with path top-left at origin using tempPaint.
809         SkMatrix translate;
810         translate.setTranslate(offset.fX, offset.fY);
811         am.set(context, translate);
812         context->drawPath(tempPaint, devPath, stroke);
813 
814         // If we're doing a normal blur, we can clobber the pathTexture in the
815         // gaussianBlur.  Otherwise, we need to save it for later compositing.
816         bool isNormalBlur = blurType == SkMaskFilter::kNormal_BlurType;
817         blurTexture.reset(context->gaussianBlur(pathTexture, isNormalBlur,
818                                                 srcRect, sigma, sigma));
819         if (NULL == blurTexture) {
820             return false;
821         }
822 
823         if (!isNormalBlur) {
824             context->setIdentityMatrix();
825             GrPaint paint;
826             SkMatrix matrix;
827             matrix.setIDiv(pathTexture->width(), pathTexture->height());
828             // Blend pathTexture over blurTexture.
829             context->setRenderTarget(blurTexture->asRenderTarget());
830             paint.colorStage(0)->setEffect(
831                 GrSimpleTextureEffect::Create(pathTexture, matrix))->unref();
832             if (SkMaskFilter::kInner_BlurType == blurType) {
833                 // inner:  dst = dst * src
834                 paint.setBlendFunc(kDC_GrBlendCoeff, kZero_GrBlendCoeff);
835             } else if (SkMaskFilter::kSolid_BlurType == blurType) {
836                 // solid:  dst = src + dst - src * dst
837                 //             = (1 - dst) * src + 1 * dst
838                 paint.setBlendFunc(kIDC_GrBlendCoeff, kOne_GrBlendCoeff);
839             } else if (SkMaskFilter::kOuter_BlurType == blurType) {
840                 // outer:  dst = dst * (1 - src)
841                 //             = 0 * src + (1 - src) * dst
842                 paint.setBlendFunc(kZero_GrBlendCoeff, kISC_GrBlendCoeff);
843             }
844             context->drawRect(paint, srcRect);
845         }
846     }
847 
848     GrContext::AutoMatrix am;
849     if (!am.setIdentity(context, grp)) {
850         return false;
851     }
852 
853     static const int MASK_IDX = GrPaint::kMaxCoverageStages - 1;
854     // we assume the last mask index is available for use
855     GrAssert(!grp->isCoverageStageEnabled(MASK_IDX));
856 
857     SkMatrix matrix;
858     matrix.setTranslate(-finalRect.fLeft, -finalRect.fTop);
859     matrix.postIDiv(blurTexture->width(), blurTexture->height());
860 
861     grp->coverageStage(MASK_IDX)->reset();
862     grp->coverageStage(MASK_IDX)->setEffect(
863         GrSimpleTextureEffect::Create(blurTexture, matrix))->unref();
864     context->drawRect(*grp, finalRect);
865     return true;
866 }
867 
drawWithMaskFilter(GrContext * context,const SkPath & devPath,SkMaskFilter * filter,const SkRegion & clip,SkBounder * bounder,GrPaint * grp,SkPaint::Style style)868 bool drawWithMaskFilter(GrContext* context, const SkPath& devPath,
869                         SkMaskFilter* filter, const SkRegion& clip, SkBounder* bounder,
870                         GrPaint* grp, SkPaint::Style style) {
871     SkMask  srcM, dstM;
872 
873     if (!SkDraw::DrawToMask(devPath, &clip.getBounds(), filter, &context->getMatrix(), &srcM,
874                             SkMask::kComputeBoundsAndRenderImage_CreateMode, style)) {
875         return false;
876     }
877     SkAutoMaskFreeImage autoSrc(srcM.fImage);
878 
879     if (!filter->filterMask(&dstM, srcM, context->getMatrix(), NULL)) {
880         return false;
881     }
882     // this will free-up dstM when we're done (allocated in filterMask())
883     SkAutoMaskFreeImage autoDst(dstM.fImage);
884 
885     if (clip.quickReject(dstM.fBounds)) {
886         return false;
887     }
888     if (bounder && !bounder->doIRect(dstM.fBounds)) {
889         return false;
890     }
891 
892     // we now have a device-aligned 8bit mask in dstM, ready to be drawn using
893     // the current clip (and identity matrix) and GrPaint settings
894     GrContext::AutoMatrix am;
895     am.setIdentity(context, grp);
896 
897     GrTextureDesc desc;
898     desc.fWidth = dstM.fBounds.width();
899     desc.fHeight = dstM.fBounds.height();
900     desc.fConfig = kAlpha_8_GrPixelConfig;
901 
902     GrAutoScratchTexture ast(context, desc);
903     GrTexture* texture = ast.texture();
904 
905     if (NULL == texture) {
906         return false;
907     }
908     texture->writePixels(0, 0, desc.fWidth, desc.fHeight, desc.fConfig,
909                                dstM.fImage, dstM.fRowBytes);
910 
911     static const int MASK_IDX = GrPaint::kMaxCoverageStages - 1;
912     // we assume the last mask index is available for use
913     GrAssert(!grp->isCoverageStageEnabled(MASK_IDX));
914 
915     SkMatrix m;
916     m.setTranslate(-dstM.fBounds.fLeft*SK_Scalar1, -dstM.fBounds.fTop*SK_Scalar1);
917     m.postIDiv(texture->width(), texture->height());
918 
919     grp->coverageStage(MASK_IDX)->setEffect(GrSimpleTextureEffect::Create(texture, m))->unref();
920     GrRect d;
921     d.setLTRB(SkIntToScalar(dstM.fBounds.fLeft),
922               SkIntToScalar(dstM.fBounds.fTop),
923               SkIntToScalar(dstM.fBounds.fRight),
924               SkIntToScalar(dstM.fBounds.fBottom));
925 
926     context->drawRect(*grp, d);
927     return true;
928 }
929 
930 }
931 
932 ///////////////////////////////////////////////////////////////////////////////
933 
drawPath(const SkDraw & draw,const SkPath & origSrcPath,const SkPaint & paint,const SkMatrix * prePathMatrix,bool pathIsMutable)934 void SkGpuDevice::drawPath(const SkDraw& draw, const SkPath& origSrcPath,
935                            const SkPaint& paint, const SkMatrix* prePathMatrix,
936                            bool pathIsMutable) {
937     CHECK_FOR_NODRAW_ANNOTATION(paint);
938     CHECK_SHOULD_DRAW(draw, false);
939 
940     GrPaint grPaint;
941     if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
942         return;
943     }
944 
945     // can we cheat, and treat a thin stroke as a hairline w/ coverage
946     // if we can, we draw lots faster (raster device does this same test)
947     SkScalar hairlineCoverage;
948     bool doHairLine = SkDrawTreatAsHairline(paint, fContext->getMatrix(), &hairlineCoverage);
949     if (doHairLine) {
950         grPaint.setCoverage(SkScalarRoundToInt(hairlineCoverage * grPaint.getCoverage()));
951     }
952 
953     // If we have a prematrix, apply it to the path, optimizing for the case
954     // where the original path can in fact be modified in place (even though
955     // its parameter type is const).
956     SkPath* pathPtr = const_cast<SkPath*>(&origSrcPath);
957     SkPath  tmpPath, effectPath;
958 
959     if (prePathMatrix) {
960         SkPath* result = pathPtr;
961 
962         if (!pathIsMutable) {
963             result = &tmpPath;
964             pathIsMutable = true;
965         }
966         // should I push prePathMatrix on our MV stack temporarily, instead
967         // of applying it here? See SkDraw.cpp
968         pathPtr->transform(*prePathMatrix, result);
969         pathPtr = result;
970     }
971     // at this point we're done with prePathMatrix
972     SkDEBUGCODE(prePathMatrix = (const SkMatrix*)0x50FF8001;)
973 
974     SkStrokeRec stroke(paint);
975     SkPathEffect* pathEffect = paint.getPathEffect();
976     const SkRect* cullRect = NULL;  // TODO: what is our bounds?
977     if (pathEffect && pathEffect->filterPath(&effectPath, *pathPtr, &stroke,
978                                              cullRect)) {
979         pathPtr = &effectPath;
980     }
981 
982     if (!pathEffect && doHairLine) {
983         stroke.setHairlineStyle();
984     }
985 
986     if (paint.getMaskFilter()) {
987         if (!stroke.isHairlineStyle()) {
988             if (stroke.applyToPath(&tmpPath, *pathPtr)) {
989                 pathPtr = &tmpPath;
990                 stroke.setFillStyle();
991             }
992         }
993 
994         // avoid possibly allocating a new path in transform if we can
995         SkPath* devPathPtr = pathIsMutable ? pathPtr : &tmpPath;
996 
997         // transform the path into device space
998         pathPtr->transform(fContext->getMatrix(), devPathPtr);
999         if (!drawWithGPUMaskFilter(fContext, *devPathPtr, stroke, paint.getMaskFilter(),
1000                                    *draw.fClip, draw.fBounder, &grPaint)) {
1001             SkPaint::Style style = stroke.isHairlineStyle() ? SkPaint::kStroke_Style :
1002                                                               SkPaint::kFill_Style;
1003             drawWithMaskFilter(fContext, *devPathPtr, paint.getMaskFilter(),
1004                                *draw.fClip, draw.fBounder, &grPaint, style);
1005         }
1006         return;
1007     }
1008 
1009     fContext->drawPath(grPaint, *pathPtr, stroke);
1010 }
1011 
1012 namespace {
1013 
get_tile_count(int l,int t,int r,int b,int tileSize)1014 inline int get_tile_count(int l, int t, int r, int b, int tileSize)  {
1015     int tilesX = (r / tileSize) - (l / tileSize) + 1;
1016     int tilesY = (b / tileSize) - (t / tileSize) + 1;
1017     return tilesX * tilesY;
1018 }
1019 
determine_tile_size(const SkBitmap & bitmap,const SkRect & src,int maxTextureSize)1020 inline int determine_tile_size(const SkBitmap& bitmap,
1021                                const SkRect& src,
1022                                int maxTextureSize) {
1023     static const int kSmallTileSize = 1 << 10;
1024     if (maxTextureSize <= kSmallTileSize) {
1025         return maxTextureSize;
1026     }
1027 
1028     size_t maxTexTotalTileSize;
1029     size_t smallTotalTileSize;
1030 
1031     SkIRect iSrc;
1032     src.roundOut(&iSrc);
1033 
1034     maxTexTotalTileSize = get_tile_count(iSrc.fLeft,
1035                                          iSrc.fTop,
1036                                          iSrc.fRight,
1037                                          iSrc.fBottom,
1038                                          maxTextureSize);
1039     smallTotalTileSize = get_tile_count(iSrc.fLeft,
1040                                         iSrc.fTop,
1041                                         iSrc.fRight,
1042                                         iSrc.fBottom,
1043                                         kSmallTileSize);
1044 
1045     maxTexTotalTileSize *= maxTextureSize * maxTextureSize;
1046     smallTotalTileSize *= kSmallTileSize * kSmallTileSize;
1047 
1048     if (maxTexTotalTileSize > 2 * smallTotalTileSize) {
1049         return kSmallTileSize;
1050     } else {
1051         return maxTextureSize;
1052     }
1053 }
1054 }
1055 
shouldTileBitmap(const SkBitmap & bitmap,const GrTextureParams & params,const SkRect * srcRectPtr) const1056 bool SkGpuDevice::shouldTileBitmap(const SkBitmap& bitmap,
1057                                    const GrTextureParams& params,
1058                                    const SkRect* srcRectPtr) const {
1059     // if bitmap is explictly texture backed then just use the texture
1060     if (NULL != bitmap.getTexture()) {
1061         return false;
1062     }
1063     // if it's larger than the max texture size, then we have no choice but
1064     // tiling
1065     const int maxTextureSize = fContext->getMaxTextureSize();
1066     if (bitmap.width() > maxTextureSize ||
1067         bitmap.height() > maxTextureSize) {
1068         return true;
1069     }
1070     // if we are going to have to draw the whole thing, then don't tile
1071     if (NULL == srcRectPtr) {
1072         return false;
1073     }
1074     // if the entire texture is already in our cache then no reason to tile it
1075     if (GrIsBitmapInCache(fContext, bitmap, &params)) {
1076         return false;
1077     }
1078 
1079     // At this point we know we could do the draw by uploading the entire bitmap
1080     // as a texture. However, if the texture would be large compared to the
1081     // cache size and we don't require most of it for this draw then tile to
1082     // reduce the amount of upload and cache spill.
1083 
1084     // assumption here is that sw bitmap size is a good proxy for its size as
1085     // a texture
1086     size_t bmpSize = bitmap.getSize();
1087     size_t cacheSize;
1088     fContext->getTextureCacheLimits(NULL, &cacheSize);
1089     if (bmpSize < cacheSize / 2) {
1090         return false;
1091     }
1092 
1093     SkScalar fracUsed = SkScalarMul(srcRectPtr->width() / bitmap.width(),
1094                                     srcRectPtr->height() / bitmap.height());
1095     if (fracUsed <= SK_ScalarHalf) {
1096         return true;
1097     } else {
1098         return false;
1099     }
1100 }
1101 
drawBitmap(const SkDraw & draw,const SkBitmap & bitmap,const SkIRect * srcRectPtr,const SkMatrix & m,const SkPaint & paint)1102 void SkGpuDevice::drawBitmap(const SkDraw& draw,
1103                              const SkBitmap& bitmap,
1104                              const SkIRect* srcRectPtr,
1105                              const SkMatrix& m,
1106                              const SkPaint& paint) {
1107 
1108     SkRect  tmp;
1109     SkRect* tmpPtr = NULL;
1110 
1111     // convert from SkIRect to SkRect
1112     if (NULL != srcRectPtr) {
1113         tmp.set(*srcRectPtr);
1114         tmpPtr = &tmp;
1115     }
1116 
1117     // We cannot call drawBitmapRect here since 'm' could be anything
1118     this->drawBitmapCommon(draw, bitmap, tmpPtr, m, paint);
1119 }
1120 
drawBitmapCommon(const SkDraw & draw,const SkBitmap & bitmap,const SkRect * srcRectPtr,const SkMatrix & m,const SkPaint & paint)1121 void SkGpuDevice::drawBitmapCommon(const SkDraw& draw,
1122                                    const SkBitmap& bitmap,
1123                                    const SkRect* srcRectPtr,
1124                                    const SkMatrix& m,
1125                                    const SkPaint& paint) {
1126     CHECK_SHOULD_DRAW(draw, false);
1127 
1128     SkRect srcRect;
1129     if (NULL == srcRectPtr) {
1130         srcRect.set(0, 0, SkIntToScalar(bitmap.width()), SkIntToScalar(bitmap.height()));
1131     } else {
1132         srcRect = *srcRectPtr;
1133     }
1134 
1135     if (paint.getMaskFilter()){
1136         // Convert the bitmap to a shader so that the rect can be drawn
1137         // through drawRect, which supports mask filters.
1138         SkMatrix        newM(m);
1139         SkBitmap        tmp;    // subset of bitmap, if necessary
1140         const SkBitmap* bitmapPtr = &bitmap;
1141         if (NULL != srcRectPtr) {
1142             SkIRect iSrc;
1143             srcRect.roundOut(&iSrc);
1144             if (!bitmap.extractSubset(&tmp, iSrc)) {
1145                 return;     // extraction failed
1146             }
1147             bitmapPtr = &tmp;
1148             srcRect.offset(SkIntToScalar(-iSrc.fLeft), SkIntToScalar(-iSrc.fTop));
1149             // The source rect has changed so update the matrix
1150             newM.preTranslate(SkIntToScalar(iSrc.fLeft), SkIntToScalar(iSrc.fTop));
1151         }
1152 
1153         SkPaint paintWithTexture(paint);
1154         paintWithTexture.setShader(SkShader::CreateBitmapShader(*bitmapPtr,
1155             SkShader::kClamp_TileMode, SkShader::kClamp_TileMode))->unref();
1156 
1157         // Transform 'newM' needs to be concatenated to the current matrix,
1158         // rather than transforming the primitive directly, so that 'newM' will
1159         // also affect the behavior of the mask filter.
1160         SkMatrix drawMatrix;
1161         drawMatrix.setConcat(fContext->getMatrix(), newM);
1162         SkDraw transformedDraw(draw);
1163         transformedDraw.fMatrix = &drawMatrix;
1164 
1165         this->drawRect(transformedDraw, srcRect, paintWithTexture);
1166 
1167         return;
1168     }
1169 
1170     GrPaint grPaint;
1171 
1172     bool alphaOnly = !(SkBitmap::kA8_Config == bitmap.config());
1173     if (!skPaint2GrPaintNoShader(this, paint, alphaOnly, false, &grPaint)) {
1174         return;
1175     }
1176     GrTextureParams params;
1177     params.setBilerp(paint.isFilterBitmap());
1178 
1179     if (!this->shouldTileBitmap(bitmap, params, srcRectPtr)) {
1180         // take the simple case
1181         this->internalDrawBitmap(bitmap, srcRect, m, params, &grPaint);
1182     } else {
1183         this->drawTiledBitmap(bitmap, srcRect, m, params, &grPaint);
1184     }
1185 }
1186 
1187 // Break 'bitmap' into several tiles to draw it since it has already
1188 // been determined to be too large to fit in VRAM
drawTiledBitmap(const SkBitmap & bitmap,const SkRect & srcRect,const SkMatrix & m,const GrTextureParams & params,GrPaint * grPaint)1189 void SkGpuDevice::drawTiledBitmap(const SkBitmap& bitmap,
1190                                   const SkRect& srcRect,
1191                                   const SkMatrix& m,
1192                                   const GrTextureParams& params,
1193                                   GrPaint* grPaint) {
1194     const int maxTextureSize = fContext->getMaxTextureSize();
1195 
1196     int tileSize = determine_tile_size(bitmap, srcRect, maxTextureSize);
1197 
1198     // compute clip bounds in local coordinates
1199     SkRect clipRect;
1200     {
1201         const GrRenderTarget* rt = fContext->getRenderTarget();
1202         clipRect.setWH(SkIntToScalar(rt->width()), SkIntToScalar(rt->height()));
1203         if (!fContext->getClip()->fClipStack->intersectRectWithClip(&clipRect)) {
1204             return;
1205         }
1206         SkMatrix matrix, inverse;
1207         matrix.setConcat(fContext->getMatrix(), m);
1208         if (!matrix.invert(&inverse)) {
1209             return;
1210         }
1211         inverse.mapRect(&clipRect);
1212     }
1213 
1214     int nx = bitmap.width() / tileSize;
1215     int ny = bitmap.height() / tileSize;
1216     for (int x = 0; x <= nx; x++) {
1217         for (int y = 0; y <= ny; y++) {
1218             SkRect tileR;
1219             tileR.set(SkIntToScalar(x * tileSize),
1220                       SkIntToScalar(y * tileSize),
1221                       SkIntToScalar((x + 1) * tileSize),
1222                       SkIntToScalar((y + 1) * tileSize));
1223 
1224             if (!SkRect::Intersects(tileR, clipRect)) {
1225                 continue;
1226             }
1227 
1228             if (!tileR.intersect(srcRect)) {
1229                 continue;
1230             }
1231 
1232             SkBitmap tmpB;
1233             SkIRect iTileR;
1234             tileR.roundOut(&iTileR);
1235             if (bitmap.extractSubset(&tmpB, iTileR)) {
1236                 // now offset it to make it "local" to our tmp bitmap
1237                 tileR.offset(SkIntToScalar(-iTileR.fLeft), SkIntToScalar(-iTileR.fTop));
1238                 SkMatrix tmpM(m);
1239                 tmpM.preTranslate(SkIntToScalar(iTileR.fLeft),
1240                                   SkIntToScalar(iTileR.fTop));
1241 
1242                 this->internalDrawBitmap(tmpB, tileR, tmpM, params, grPaint);
1243             }
1244         }
1245     }
1246 }
1247 
1248 namespace {
1249 
hasAlignedSamples(const SkRect & srcRect,const SkRect & transformedRect)1250 bool hasAlignedSamples(const SkRect& srcRect, const SkRect& transformedRect) {
1251     // detect pixel disalignment
1252     if (SkScalarAbs(SkScalarRoundToScalar(transformedRect.left()) -
1253             transformedRect.left()) < COLOR_BLEED_TOLERANCE &&
1254         SkScalarAbs(SkScalarRoundToScalar(transformedRect.top()) -
1255             transformedRect.top()) < COLOR_BLEED_TOLERANCE &&
1256         SkScalarAbs(transformedRect.width() - srcRect.width()) <
1257             COLOR_BLEED_TOLERANCE &&
1258         SkScalarAbs(transformedRect.height() - srcRect.height()) <
1259             COLOR_BLEED_TOLERANCE) {
1260         return true;
1261     }
1262     return false;
1263 }
1264 
mayColorBleed(const SkRect & srcRect,const SkRect & transformedRect,const SkMatrix & m)1265 bool mayColorBleed(const SkRect& srcRect, const SkRect& transformedRect,
1266                    const SkMatrix& m) {
1267     // Only gets called if hasAlignedSamples returned false.
1268     // So we can assume that sampling is axis aligned but not texel aligned.
1269     GrAssert(!hasAlignedSamples(srcRect, transformedRect));
1270     SkRect innerSrcRect(srcRect), innerTransformedRect,
1271         outerTransformedRect(transformedRect);
1272     innerSrcRect.inset(SK_ScalarHalf, SK_ScalarHalf);
1273     m.mapRect(&innerTransformedRect, innerSrcRect);
1274 
1275     // The gap between outerTransformedRect and innerTransformedRect
1276     // represents the projection of the source border area, which is
1277     // problematic for color bleeding.  We must check whether any
1278     // destination pixels sample the border area.
1279     outerTransformedRect.inset(COLOR_BLEED_TOLERANCE, COLOR_BLEED_TOLERANCE);
1280     innerTransformedRect.outset(COLOR_BLEED_TOLERANCE, COLOR_BLEED_TOLERANCE);
1281     SkIRect outer, inner;
1282     outerTransformedRect.round(&outer);
1283     innerTransformedRect.round(&inner);
1284     // If the inner and outer rects round to the same result, it means the
1285     // border does not overlap any pixel centers. Yay!
1286     return inner != outer;
1287 }
1288 
1289 } // unnamed namespace
1290 
1291 /*
1292  *  This is called by drawBitmap(), which has to handle images that may be too
1293  *  large to be represented by a single texture.
1294  *
1295  *  internalDrawBitmap assumes that the specified bitmap will fit in a texture
1296  *  and that non-texture portion of the GrPaint has already been setup.
1297  */
internalDrawBitmap(const SkBitmap & bitmap,const SkRect & srcRect,const SkMatrix & m,const GrTextureParams & params,GrPaint * grPaint)1298 void SkGpuDevice::internalDrawBitmap(const SkBitmap& bitmap,
1299                                      const SkRect& srcRect,
1300                                      const SkMatrix& m,
1301                                      const GrTextureParams& params,
1302                                      GrPaint* grPaint) {
1303     SkASSERT(bitmap.width() <= fContext->getMaxTextureSize() &&
1304              bitmap.height() <= fContext->getMaxTextureSize());
1305 
1306     SkAutoLockPixels alp(bitmap, !bitmap.getTexture());
1307     if (!bitmap.getTexture() && !bitmap.readyToDraw()) {
1308         SkDebugf("nothing to draw\n");
1309         return;
1310     }
1311 
1312     GrTexture* texture;
1313     SkAutoCachedTexture act(this, bitmap, &params, &texture);
1314     if (NULL == texture) {
1315         return;
1316     }
1317 
1318     GrRect dstRect(srcRect);
1319     GrRect paintRect;
1320     SkScalar wInv = SkScalarInvert(SkIntToScalar(bitmap.width()));
1321     SkScalar hInv = SkScalarInvert(SkIntToScalar(bitmap.height()));
1322     paintRect.setLTRB(SkScalarMul(srcRect.fLeft,   wInv),
1323                       SkScalarMul(srcRect.fTop,    hInv),
1324                       SkScalarMul(srcRect.fRight,  wInv),
1325                       SkScalarMul(srcRect.fBottom, hInv));
1326 
1327     bool needsTextureDomain = false;
1328     if (params.isBilerp()) {
1329         // Need texture domain if drawing a sub rect.
1330         needsTextureDomain = srcRect.width() < bitmap.width() ||
1331                              srcRect.height() < bitmap.height();
1332         if (m.rectStaysRect() && fContext->getMatrix().rectStaysRect()) {
1333             // sampling is axis-aligned
1334             GrRect transformedRect;
1335             SkMatrix srcToDeviceMatrix(m);
1336             srcToDeviceMatrix.postConcat(fContext->getMatrix());
1337             srcToDeviceMatrix.mapRect(&transformedRect, srcRect);
1338 
1339             if (hasAlignedSamples(srcRect, transformedRect)) {
1340                 // We could also turn off filtering here (but we already did a cache lookup with
1341                 // params).
1342                 needsTextureDomain = false;
1343             } else {
1344                 needsTextureDomain = needsTextureDomain &&
1345                     mayColorBleed(srcRect, transformedRect, m);
1346             }
1347         }
1348     }
1349 
1350     GrRect textureDomain = GrRect::MakeEmpty();
1351     SkAutoTUnref<GrEffectRef> effect;
1352     if (needsTextureDomain) {
1353         // Use a constrained texture domain to avoid color bleeding
1354         SkScalar left, top, right, bottom;
1355         if (srcRect.width() > SK_Scalar1) {
1356             SkScalar border = SK_ScalarHalf / bitmap.width();
1357             left = paintRect.left() + border;
1358             right = paintRect.right() - border;
1359         } else {
1360             left = right = SkScalarHalf(paintRect.left() + paintRect.right());
1361         }
1362         if (srcRect.height() > SK_Scalar1) {
1363             SkScalar border = SK_ScalarHalf / bitmap.height();
1364             top = paintRect.top() + border;
1365             bottom = paintRect.bottom() - border;
1366         } else {
1367             top = bottom = SkScalarHalf(paintRect.top() + paintRect.bottom());
1368         }
1369         textureDomain.setLTRB(left, top, right, bottom);
1370         effect.reset(GrTextureDomainEffect::Create(texture,
1371                                                    SkMatrix::I(),
1372                                                    textureDomain,
1373                                                    GrTextureDomainEffect::kClamp_WrapMode,
1374                                                    params.isBilerp()));
1375     } else {
1376         effect.reset(GrSimpleTextureEffect::Create(texture, SkMatrix::I(), params));
1377     }
1378     grPaint->colorStage(kBitmapTextureIdx)->setEffect(effect);
1379     fContext->drawRectToRect(*grPaint, dstRect, paintRect, &m);
1380 }
1381 
1382 namespace {
1383 
apply_effect(GrContext * context,GrTexture * srcTexture,GrTexture * dstTexture,const GrRect & rect,GrEffectRef * effect)1384 void apply_effect(GrContext* context,
1385                   GrTexture* srcTexture,
1386                   GrTexture* dstTexture,
1387                   const GrRect& rect,
1388                   GrEffectRef* effect) {
1389     SkASSERT(srcTexture && srcTexture->getContext() == context);
1390     GrContext::AutoMatrix am;
1391     am.setIdentity(context);
1392     GrContext::AutoRenderTarget art(context, dstTexture->asRenderTarget());
1393     GrContext::AutoClip acs(context, rect);
1394 
1395     GrPaint paint;
1396     paint.colorStage(0)->setEffect(effect);
1397     context->drawRect(paint, rect);
1398 }
1399 
1400 };
1401 
wrap_texture(GrTexture * texture)1402 static SkBitmap wrap_texture(GrTexture* texture) {
1403     SkBitmap result;
1404     bool dummy;
1405     SkBitmap::Config config = grConfig2skConfig(texture->config(), &dummy);
1406     result.setConfig(config, texture->width(), texture->height());
1407     result.setPixelRef(SkNEW_ARGS(SkGrPixelRef, (texture)))->unref();
1408     return result;
1409 }
1410 
filter_texture(SkDevice * device,GrContext * context,GrTexture * texture,SkImageFilter * filter,int w,int h,SkBitmap * result)1411 static bool filter_texture(SkDevice* device, GrContext* context,
1412                            GrTexture* texture, SkImageFilter* filter,
1413                            int w, int h, SkBitmap* result) {
1414     GrAssert(filter);
1415     SkDeviceImageFilterProxy proxy(device);
1416 
1417     GrTextureDesc desc;
1418     desc.fFlags = kRenderTarget_GrTextureFlagBit,
1419     desc.fWidth = w;
1420     desc.fHeight = h;
1421     desc.fConfig = kRGBA_8888_GrPixelConfig;
1422     GrEffectRef* effect;
1423 
1424     if (filter->canFilterImageGPU()) {
1425         // Save the render target and set it to NULL, so we don't accidentally draw to it in the
1426         // filter.  Also set the clip wide open and the matrix to identity.
1427         GrContext::AutoWideOpenIdentityDraw awo(context, NULL);
1428         return filter->filterImageGPU(&proxy, wrap_texture(texture), result);
1429     } else if (filter->asNewEffect(&effect, texture)) {
1430         GrAutoScratchTexture dst(context, desc);
1431         SkRect r = SkRect::MakeWH(SkIntToScalar(w), SkIntToScalar(h));
1432         apply_effect(context, texture, dst.texture(), r, effect);
1433         SkAutoTUnref<GrTexture> resultTex(dst.detach());
1434         effect->unref();
1435         *result = wrap_texture(resultTex.get());
1436         return true;
1437     } else {
1438         return false;
1439     }
1440 }
1441 
drawSprite(const SkDraw & draw,const SkBitmap & bitmap,int left,int top,const SkPaint & paint)1442 void SkGpuDevice::drawSprite(const SkDraw& draw, const SkBitmap& bitmap,
1443                              int left, int top, const SkPaint& paint) {
1444     // drawSprite is defined to be in device coords.
1445     CHECK_SHOULD_DRAW(draw, true);
1446 
1447     SkAutoLockPixels alp(bitmap, !bitmap.getTexture());
1448     if (!bitmap.getTexture() && !bitmap.readyToDraw()) {
1449         return;
1450     }
1451 
1452     int w = bitmap.width();
1453     int h = bitmap.height();
1454 
1455     GrPaint grPaint;
1456     if(!skPaint2GrPaintNoShader(this, paint, true, false, &grPaint)) {
1457         return;
1458     }
1459 
1460     GrEffectStage* stage = grPaint.colorStage(kBitmapTextureIdx);
1461 
1462     GrTexture* texture;
1463     stage->reset();
1464     // draw sprite uses the default texture params
1465     SkAutoCachedTexture act(this, bitmap, NULL, &texture);
1466     grPaint.colorStage(kBitmapTextureIdx)->setEffect(
1467         GrSimpleTextureEffect::Create(texture, SkMatrix::I()))->unref();
1468 
1469     SkImageFilter* filter = paint.getImageFilter();
1470     if (NULL != filter) {
1471         SkBitmap filterBitmap;
1472         if (filter_texture(this, fContext, texture, filter, w, h, &filterBitmap)) {
1473             grPaint.colorStage(kBitmapTextureIdx)->setEffect(
1474                 GrSimpleTextureEffect::Create((GrTexture*) filterBitmap.getTexture(), SkMatrix::I()))->unref();
1475             texture = (GrTexture*) filterBitmap.getTexture();
1476             w = filterBitmap.width();
1477             h = filterBitmap.height();
1478         }
1479     }
1480 
1481     fContext->drawRectToRect(grPaint,
1482                             GrRect::MakeXYWH(SkIntToScalar(left),
1483                                             SkIntToScalar(top),
1484                                             SkIntToScalar(w),
1485                                             SkIntToScalar(h)),
1486                             GrRect::MakeWH(SK_Scalar1 * w / texture->width(),
1487                                         SK_Scalar1 * h / texture->height()));
1488 }
1489 
drawBitmapRect(const SkDraw & draw,const SkBitmap & bitmap,const SkRect * src,const SkRect & dst,const SkPaint & paint)1490 void SkGpuDevice::drawBitmapRect(const SkDraw& draw, const SkBitmap& bitmap,
1491                                  const SkRect* src, const SkRect& dst,
1492                                  const SkPaint& paint) {
1493     SkMatrix    matrix;
1494     SkRect      bitmapBounds, tmpSrc;
1495 
1496     bitmapBounds.set(0, 0,
1497                      SkIntToScalar(bitmap.width()),
1498                      SkIntToScalar(bitmap.height()));
1499 
1500     // Compute matrix from the two rectangles
1501     if (NULL != src) {
1502         tmpSrc = *src;
1503     } else {
1504         tmpSrc = bitmapBounds;
1505     }
1506     matrix.setRectToRect(tmpSrc, dst, SkMatrix::kFill_ScaleToFit);
1507 
1508     // clip the tmpSrc to the bounds of the bitmap. No check needed if src==null.
1509     if (NULL != src) {
1510         if (!bitmapBounds.contains(tmpSrc)) {
1511             if (!tmpSrc.intersect(bitmapBounds)) {
1512                 return; // nothing to draw
1513             }
1514         }
1515     }
1516 
1517     this->drawBitmapCommon(draw, bitmap, &tmpSrc, matrix, paint);
1518 }
1519 
drawDevice(const SkDraw & draw,SkDevice * device,int x,int y,const SkPaint & paint)1520 void SkGpuDevice::drawDevice(const SkDraw& draw, SkDevice* device,
1521                              int x, int y, const SkPaint& paint) {
1522     // clear of the source device must occur before CHECK_SHOULD_DRAW
1523     SkGpuDevice* dev = static_cast<SkGpuDevice*>(device);
1524     if (dev->fNeedClear) {
1525         // TODO: could check here whether we really need to draw at all
1526         dev->clear(0x0);
1527     }
1528 
1529     // drawDevice is defined to be in device coords.
1530     CHECK_SHOULD_DRAW(draw, true);
1531 
1532     GrPaint grPaint;
1533     grPaint.colorStage(kBitmapTextureIdx)->reset();
1534     if (!dev->bindDeviceAsTexture(&grPaint) ||
1535         !skPaint2GrPaintNoShader(this, paint, true, false, &grPaint)) {
1536         return;
1537     }
1538 
1539     GrTexture* devTex = (*grPaint.getColorStage(kBitmapTextureIdx).getEffect())->texture(0);
1540     SkASSERT(NULL != devTex);
1541 
1542     const SkBitmap& bm = dev->accessBitmap(false);
1543     int w = bm.width();
1544     int h = bm.height();
1545 
1546     SkImageFilter* filter = paint.getImageFilter();
1547     if (NULL != filter) {
1548         SkBitmap filterBitmap;
1549         if (filter_texture(this, fContext, devTex, filter, w, h, &filterBitmap)) {
1550             grPaint.colorStage(kBitmapTextureIdx)->setEffect(
1551                 GrSimpleTextureEffect::Create((GrTexture*) filterBitmap.getTexture(), SkMatrix::I()))->unref();
1552             devTex = (GrTexture*) filterBitmap.getTexture();
1553             w = filterBitmap.width();
1554             h = filterBitmap.height();
1555         }
1556     }
1557 
1558     GrRect dstRect = GrRect::MakeXYWH(SkIntToScalar(x),
1559                                       SkIntToScalar(y),
1560                                       SkIntToScalar(w),
1561                                       SkIntToScalar(h));
1562 
1563     // The device being drawn may not fill up its texture (saveLayer uses
1564     // the approximate ).
1565     GrRect srcRect = GrRect::MakeWH(SK_Scalar1 * w / devTex->width(),
1566                                     SK_Scalar1 * h / devTex->height());
1567 
1568     fContext->drawRectToRect(grPaint, dstRect, srcRect);
1569 }
1570 
canHandleImageFilter(SkImageFilter * filter)1571 bool SkGpuDevice::canHandleImageFilter(SkImageFilter* filter) {
1572     if (!filter->asNewEffect(NULL, NULL) &&
1573         !filter->canFilterImageGPU()) {
1574         return false;
1575     }
1576     return true;
1577 }
1578 
filterImage(SkImageFilter * filter,const SkBitmap & src,const SkMatrix & ctm,SkBitmap * result,SkIPoint * offset)1579 bool SkGpuDevice::filterImage(SkImageFilter* filter, const SkBitmap& src,
1580                               const SkMatrix& ctm,
1581                               SkBitmap* result, SkIPoint* offset) {
1582     // want explicitly our impl, so guard against a subclass of us overriding it
1583     if (!this->SkGpuDevice::canHandleImageFilter(filter)) {
1584         return false;
1585     }
1586 
1587     SkAutoLockPixels alp(src, !src.getTexture());
1588     if (!src.getTexture() && !src.readyToDraw()) {
1589         return false;
1590     }
1591 
1592     GrPaint paint;
1593 
1594     GrTexture* texture;
1595     // We assume here that the filter will not attempt to tile the src. Otherwise, this cache lookup
1596     // must be pushed upstack.
1597     SkAutoCachedTexture act(this, src, NULL, &texture);
1598 
1599     return filter_texture(this, fContext, texture, filter, src.width(), src.height(), result);
1600 }
1601 
1602 ///////////////////////////////////////////////////////////////////////////////
1603 
1604 // must be in SkCanvas::VertexMode order
1605 static const GrPrimitiveType gVertexMode2PrimitiveType[] = {
1606     kTriangles_GrPrimitiveType,
1607     kTriangleStrip_GrPrimitiveType,
1608     kTriangleFan_GrPrimitiveType,
1609 };
1610 
drawVertices(const SkDraw & draw,SkCanvas::VertexMode vmode,int vertexCount,const SkPoint vertices[],const SkPoint texs[],const SkColor colors[],SkXfermode * xmode,const uint16_t indices[],int indexCount,const SkPaint & paint)1611 void SkGpuDevice::drawVertices(const SkDraw& draw, SkCanvas::VertexMode vmode,
1612                               int vertexCount, const SkPoint vertices[],
1613                               const SkPoint texs[], const SkColor colors[],
1614                               SkXfermode* xmode,
1615                               const uint16_t indices[], int indexCount,
1616                               const SkPaint& paint) {
1617     CHECK_SHOULD_DRAW(draw, false);
1618 
1619     GrPaint grPaint;
1620     // we ignore the shader if texs is null.
1621     if (NULL == texs) {
1622         if (!skPaint2GrPaintNoShader(this, paint, false, NULL == colors, &grPaint)) {
1623             return;
1624         }
1625     } else {
1626         if (!skPaint2GrPaintShader(this, paint, NULL == colors, &grPaint)) {
1627             return;
1628         }
1629     }
1630 
1631     if (NULL != xmode && NULL != texs && NULL != colors) {
1632         if (!SkXfermode::IsMode(xmode, SkXfermode::kModulate_Mode)) {
1633             SkDebugf("Unsupported vertex-color/texture xfer mode.\n");
1634 #if 0
1635             return
1636 #endif
1637         }
1638     }
1639 
1640     SkAutoSTMalloc<128, GrColor> convertedColors(0);
1641     if (NULL != colors) {
1642         // need to convert byte order and from non-PM to PM
1643         convertedColors.reset(vertexCount);
1644         for (int i = 0; i < vertexCount; ++i) {
1645             convertedColors[i] = SkColor2GrColor(colors[i]);
1646         }
1647         colors = convertedColors.get();
1648     }
1649     fContext->drawVertices(grPaint,
1650                            gVertexMode2PrimitiveType[vmode],
1651                            vertexCount,
1652                            (GrPoint*) vertices,
1653                            (GrPoint*) texs,
1654                            colors,
1655                            indices,
1656                            indexCount);
1657 }
1658 
1659 ///////////////////////////////////////////////////////////////////////////////
1660 
GlyphCacheAuxProc(void * data)1661 static void GlyphCacheAuxProc(void* data) {
1662     GrFontScaler* scaler = (GrFontScaler*)data;
1663     SkSafeUnref(scaler);
1664 }
1665 
get_gr_font_scaler(SkGlyphCache * cache)1666 static GrFontScaler* get_gr_font_scaler(SkGlyphCache* cache) {
1667     void* auxData;
1668     GrFontScaler* scaler = NULL;
1669     if (cache->getAuxProcData(GlyphCacheAuxProc, &auxData)) {
1670         scaler = (GrFontScaler*)auxData;
1671     }
1672     if (NULL == scaler) {
1673         scaler = SkNEW_ARGS(SkGrFontScaler, (cache));
1674         cache->setAuxProc(GlyphCacheAuxProc, scaler);
1675     }
1676     return scaler;
1677 }
1678 
SkGPU_Draw1Glyph(const SkDraw1Glyph & state,SkFixed fx,SkFixed fy,const SkGlyph & glyph)1679 static void SkGPU_Draw1Glyph(const SkDraw1Glyph& state,
1680                              SkFixed fx, SkFixed fy,
1681                              const SkGlyph& glyph) {
1682     SkASSERT(glyph.fWidth > 0 && glyph.fHeight > 0);
1683 
1684     GrSkDrawProcs* procs = static_cast<GrSkDrawProcs*>(state.fDraw->fProcs);
1685 
1686     if (NULL == procs->fFontScaler) {
1687         procs->fFontScaler = get_gr_font_scaler(state.fCache);
1688     }
1689 
1690     procs->fTextContext->drawPackedGlyph(GrGlyph::Pack(glyph.getGlyphID(),
1691                                                        glyph.getSubXFixed(),
1692                                                        glyph.getSubYFixed()),
1693                                          SkFixedFloorToFixed(fx),
1694                                          SkFixedFloorToFixed(fy),
1695                                          procs->fFontScaler);
1696 }
1697 
initDrawForText(GrTextContext * context)1698 SkDrawProcs* SkGpuDevice::initDrawForText(GrTextContext* context) {
1699 
1700     // deferred allocation
1701     if (NULL == fDrawProcs) {
1702         fDrawProcs = SkNEW(GrSkDrawProcs);
1703         fDrawProcs->fD1GProc = SkGPU_Draw1Glyph;
1704         fDrawProcs->fContext = fContext;
1705     }
1706 
1707     // init our (and GL's) state
1708     fDrawProcs->fTextContext = context;
1709     fDrawProcs->fFontScaler = NULL;
1710     return fDrawProcs;
1711 }
1712 
drawText(const SkDraw & draw,const void * text,size_t byteLength,SkScalar x,SkScalar y,const SkPaint & paint)1713 void SkGpuDevice::drawText(const SkDraw& draw, const void* text,
1714                           size_t byteLength, SkScalar x, SkScalar y,
1715                           const SkPaint& paint) {
1716     CHECK_SHOULD_DRAW(draw, false);
1717 
1718     if (fContext->getMatrix().hasPerspective()) {
1719         // this guy will just call our drawPath()
1720         draw.drawText((const char*)text, byteLength, x, y, paint);
1721     } else {
1722         SkDraw myDraw(draw);
1723 
1724         GrPaint grPaint;
1725         if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
1726             return;
1727         }
1728         GrTextContext context(fContext, grPaint);
1729         myDraw.fProcs = this->initDrawForText(&context);
1730         this->INHERITED::drawText(myDraw, text, byteLength, x, y, paint);
1731     }
1732 }
1733 
drawPosText(const SkDraw & draw,const void * text,size_t byteLength,const SkScalar pos[],SkScalar constY,int scalarsPerPos,const SkPaint & paint)1734 void SkGpuDevice::drawPosText(const SkDraw& draw, const void* text,
1735                              size_t byteLength, const SkScalar pos[],
1736                              SkScalar constY, int scalarsPerPos,
1737                              const SkPaint& paint) {
1738     CHECK_SHOULD_DRAW(draw, false);
1739 
1740     if (fContext->getMatrix().hasPerspective()) {
1741         // this guy will just call our drawPath()
1742         draw.drawPosText((const char*)text, byteLength, pos, constY,
1743                          scalarsPerPos, paint);
1744     } else {
1745         SkDraw myDraw(draw);
1746 
1747         GrPaint grPaint;
1748         if (!skPaint2GrPaintShader(this, paint, true, &grPaint)) {
1749             return;
1750         }
1751         GrTextContext context(fContext, grPaint);
1752         myDraw.fProcs = this->initDrawForText(&context);
1753         this->INHERITED::drawPosText(myDraw, text, byteLength, pos, constY,
1754                                      scalarsPerPos, paint);
1755     }
1756 }
1757 
drawTextOnPath(const SkDraw & draw,const void * text,size_t len,const SkPath & path,const SkMatrix * m,const SkPaint & paint)1758 void SkGpuDevice::drawTextOnPath(const SkDraw& draw, const void* text,
1759                                 size_t len, const SkPath& path,
1760                                 const SkMatrix* m, const SkPaint& paint) {
1761     CHECK_SHOULD_DRAW(draw, false);
1762 
1763     SkASSERT(draw.fDevice == this);
1764     draw.drawTextOnPath((const char*)text, len, path, m, paint);
1765 }
1766 
1767 ///////////////////////////////////////////////////////////////////////////////
1768 
filterTextFlags(const SkPaint & paint,TextFlags * flags)1769 bool SkGpuDevice::filterTextFlags(const SkPaint& paint, TextFlags* flags) {
1770     if (!paint.isLCDRenderText()) {
1771         // we're cool with the paint as is
1772         return false;
1773     }
1774 
1775     if (paint.getShader() ||
1776         paint.getXfermode() || // unless its srcover
1777         paint.getMaskFilter() ||
1778         paint.getRasterizer() ||
1779         paint.getColorFilter() ||
1780         paint.getPathEffect() ||
1781         paint.isFakeBoldText() ||
1782         paint.getStyle() != SkPaint::kFill_Style) {
1783         // turn off lcd
1784         flags->fFlags = paint.getFlags() & ~SkPaint::kLCDRenderText_Flag;
1785         flags->fHinting = paint.getHinting();
1786         return true;
1787     }
1788     // we're cool with the paint as is
1789     return false;
1790 }
1791 
flush()1792 void SkGpuDevice::flush() {
1793     DO_DEFERRED_CLEAR();
1794     fContext->resolveRenderTarget(fRenderTarget);
1795 }
1796 
1797 ///////////////////////////////////////////////////////////////////////////////
1798 
onCreateCompatibleDevice(SkBitmap::Config config,int width,int height,bool isOpaque,Usage usage)1799 SkDevice* SkGpuDevice::onCreateCompatibleDevice(SkBitmap::Config config,
1800                                                 int width, int height,
1801                                                 bool isOpaque,
1802                                                 Usage usage) {
1803     GrTextureDesc desc;
1804     desc.fConfig = fRenderTarget->config();
1805     desc.fFlags = kRenderTarget_GrTextureFlagBit;
1806     desc.fWidth = width;
1807     desc.fHeight = height;
1808     desc.fSampleCnt = fRenderTarget->numSamples();
1809 
1810     SkAutoTUnref<GrTexture> texture;
1811     // Skia's convention is to only clear a device if it is non-opaque.
1812     bool needClear = !isOpaque;
1813 
1814 #if CACHE_COMPATIBLE_DEVICE_TEXTURES
1815     // layers are never draw in repeat modes, so we can request an approx
1816     // match and ignore any padding.
1817     const GrContext::ScratchTexMatch match = (kSaveLayer_Usage == usage) ?
1818                                                 GrContext::kApprox_ScratchTexMatch :
1819                                                 GrContext::kExact_ScratchTexMatch;
1820     texture.reset(fContext->lockAndRefScratchTexture(desc, match));
1821 #else
1822     texture.reset(fContext->createUncachedTexture(desc, NULL, 0));
1823 #endif
1824     if (NULL != texture.get()) {
1825         return SkNEW_ARGS(SkGpuDevice,(fContext, texture, needClear));
1826     } else {
1827         GrPrintf("---- failed to create compatible device texture [%d %d]\n", width, height);
1828         return NULL;
1829     }
1830 }
1831 
SkGpuDevice(GrContext * context,GrTexture * texture,bool needClear)1832 SkGpuDevice::SkGpuDevice(GrContext* context,
1833                          GrTexture* texture,
1834                          bool needClear)
1835     : SkDevice(make_bitmap(context, texture->asRenderTarget())) {
1836 
1837     GrAssert(texture && texture->asRenderTarget());
1838     // This constructor is called from onCreateCompatibleDevice. It has locked the RT in the texture
1839     // cache. We pass true for the third argument so that it will get unlocked.
1840     this->initFromRenderTarget(context, texture->asRenderTarget(), true);
1841     fNeedClear = needClear;
1842 }
1843