• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2     Copyright 2011 Google Inc.
3 
4     Licensed under the Apache License, Version 2.0 (the "License");
5     you may not use this file except in compliance with the License.
6     You may obtain a copy of the License at
7 
8          http://www.apache.org/licenses/LICENSE-2.0
9 
10     Unless required by applicable law or agreed to in writing, software
11     distributed under the License is distributed on an "AS IS" BASIS,
12     WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13     See the License for the specific language governing permissions and
14     limitations under the License.
15  */
16 
17 
18 #include "GrContext.h"
19 #include "GrTextContext.h"
20 
21 #include "SkGpuDevice.h"
22 #include "SkGpuDeviceFactory.h"
23 #include "SkGrTexturePixelRef.h"
24 
25 #include "SkColorFilter.h"
26 #include "SkDrawProcs.h"
27 #include "SkGlyphCache.h"
28 #include "SkUtils.h"
29 
30 #define CACHE_LAYER_TEXTURES 1
31 
32 #if 0
33     extern bool (*gShouldDrawProc)();
34     #define CHECK_SHOULD_DRAW(draw)                             \
35         do {                                                    \
36             if (gShouldDrawProc && !gShouldDrawProc()) return;  \
37             this->prepareRenderTarget(draw);                    \
38         } while (0)
39 #else
40     #define CHECK_SHOULD_DRAW(draw) this->prepareRenderTarget(draw)
41 #endif
42 
43 // we use the same texture slot on GrPaint for bitmaps and shaders
44 // (since drawBitmap, drawSprite, and drawDevice ignore skia's shader)
45 enum {
46     kBitmapTextureIdx = 0,
47     kShaderTextureIdx = 0
48 };
49 
50 ///////////////////////////////////////////////////////////////////////////////
51 
52 SkGpuDevice::SkAutoCachedTexture::
SkAutoCachedTexture(SkGpuDevice * device,const SkBitmap & bitmap,const GrSamplerState & sampler,GrTexture ** texture)53              SkAutoCachedTexture(SkGpuDevice* device,
54                                  const SkBitmap& bitmap,
55                                  const GrSamplerState& sampler,
56                                  GrTexture** texture) {
57     GrAssert(texture);
58     fTex = NULL;
59     *texture = this->set(device, bitmap, sampler);
60 }
61 
SkAutoCachedTexture()62 SkGpuDevice::SkAutoCachedTexture::SkAutoCachedTexture() {
63     fTex = NULL;
64 }
65 
set(SkGpuDevice * device,const SkBitmap & bitmap,const GrSamplerState & sampler)66 GrTexture* SkGpuDevice::SkAutoCachedTexture::set(SkGpuDevice* device,
67                                                  const SkBitmap& bitmap,
68                                                  const GrSamplerState& sampler) {
69     if (fTex) {
70         fDevice->unlockCachedTexture(fTex);
71     }
72     fDevice = device;
73     GrTexture* texture = (GrTexture*)bitmap.getTexture();
74     if (texture) {
75         // return the native texture
76         fTex = NULL;
77     } else {
78         // look it up in our cache
79         fTex = device->lockCachedTexture(bitmap, sampler, &texture, false);
80     }
81     return texture;
82 }
83 
~SkAutoCachedTexture()84 SkGpuDevice::SkAutoCachedTexture::~SkAutoCachedTexture() {
85     if (fTex) {
86         fDevice->unlockCachedTexture(fTex);
87     }
88 }
89 
90 ///////////////////////////////////////////////////////////////////////////////
91 
92 bool gDoTraceDraw;
93 
94 struct GrSkDrawProcs : public SkDrawProcs {
95 public:
96     GrContext* fContext;
97     GrTextContext* fTextContext;
98     GrFontScaler* fFontScaler;  // cached in the skia glyphcache
99 };
100 
101 ///////////////////////////////////////////////////////////////////////////////
102 
Current3DApiRenderTarget()103 GrRenderTarget* SkGpuDevice::Current3DApiRenderTarget() {
104     return (GrRenderTarget*) -1;
105 }
106 
SkGpuDevice(GrContext * context,const SkBitmap & bitmap,GrRenderTarget * renderTargetOrNull)107 SkGpuDevice::SkGpuDevice(GrContext* context,
108                          const SkBitmap& bitmap,
109                          GrRenderTarget* renderTargetOrNull)
110         : SkDevice(NULL, bitmap, (NULL == renderTargetOrNull)) {
111 
112     fNeedPrepareRenderTarget = false;
113     fDrawProcs = NULL;
114 
115     fContext = context;
116     fContext->ref();
117 
118     fCache = NULL;
119     fTexture = NULL;
120     fRenderTarget = NULL;
121     fNeedClear = false;
122 
123     if (NULL == renderTargetOrNull) {
124         SkBitmap::Config c = bitmap.config();
125         if (c != SkBitmap::kRGB_565_Config) {
126             c = SkBitmap::kARGB_8888_Config;
127         }
128         SkBitmap bm;
129         bm.setConfig(c, this->width(), this->height());
130 
131 #if CACHE_LAYER_TEXTURES
132 
133         fCache = this->lockCachedTexture(bm, GrSamplerState::ClampNoFilter(),
134                        &fTexture, true);
135         if (fCache) {
136             SkASSERT(NULL != fTexture);
137             SkASSERT(NULL != fTexture->asRenderTarget());
138         }
139 #else
140         const GrTextureDesc desc = {
141             kRenderTarget_GrTextureFlagBit,
142             kNone_GrAALevel,
143             this->width(),
144             this->height(),
145             SkGr::Bitmap2PixelConfig(bm)
146         };
147 
148         fTexture = fContext->createUncachedTexture(desc, NULL, 0);
149 #endif
150         if (NULL != fTexture) {
151             fRenderTarget = fTexture->asRenderTarget();
152 
153             GrAssert(NULL != fRenderTarget);
154 
155             // we defer the actual clear until our gainFocus()
156             fNeedClear = true;
157 
158             // wrap the bitmap with a pixelref to expose our texture
159             SkGrTexturePixelRef* pr = new SkGrTexturePixelRef(fTexture);
160             this->setPixelRef(pr, 0)->unref();
161         } else {
162             GrPrintf("--- failed to create gpu-offscreen [%d %d]\n",
163                      this->width(), this->height());
164             GrAssert(false);
165         }
166     } else {
167         if (Current3DApiRenderTarget() == renderTargetOrNull) {
168             fRenderTarget = fContext->createRenderTargetFrom3DApiState();
169         } else {
170             fRenderTarget = renderTargetOrNull;
171             fRenderTarget->ref();
172         }
173         SkGrRenderTargetPixelRef* pr = new SkGrRenderTargetPixelRef(fRenderTarget);
174         this->setPixelRef(pr, 0)->unref();
175     }
176 }
177 
~SkGpuDevice()178 SkGpuDevice::~SkGpuDevice() {
179     if (fDrawProcs) {
180         delete fDrawProcs;
181     }
182 
183     if (fCache) {
184         GrAssert(NULL != fTexture);
185         GrAssert(fRenderTarget == fTexture->asRenderTarget());
186         fContext->unlockTexture((GrTextureEntry*)fCache);
187     } else if (NULL != fTexture) {
188         GrAssert(!CACHE_LAYER_TEXTURES);
189         GrAssert(fRenderTarget == fTexture->asRenderTarget());
190         fTexture->unref();
191     } else if (NULL != fRenderTarget) {
192         fRenderTarget->unref();
193     }
194     fContext->unref();
195 }
196 
getLayerTextureHandle() const197 intptr_t SkGpuDevice::getLayerTextureHandle() const {
198     if (fTexture) {
199         return fTexture->getTextureHandle();
200     } else {
201         return 0;
202     }
203 }
204 
onNewDeviceFactory()205 SkDeviceFactory* SkGpuDevice::onNewDeviceFactory() {
206     return SkNEW_ARGS(SkGpuDeviceFactory, (fContext, fRenderTarget));
207 }
208 
209 ///////////////////////////////////////////////////////////////////////////////
210 
makeRenderTargetCurrent()211 void SkGpuDevice::makeRenderTargetCurrent() {
212     fContext->setRenderTarget(fRenderTarget);
213     fContext->flush(true);
214     fNeedPrepareRenderTarget = true;
215 }
216 
217 ///////////////////////////////////////////////////////////////////////////////
218 
readPixels(const SkIRect & srcRect,SkBitmap * bitmap)219 bool SkGpuDevice::readPixels(const SkIRect& srcRect, SkBitmap* bitmap) {
220     SkIRect bounds;
221     bounds.set(0, 0, this->width(), this->height());
222     if (!bounds.intersect(srcRect)) {
223         return false;
224     }
225 
226     const int w = bounds.width();
227     const int h = bounds.height();
228     SkBitmap tmp;
229     // note we explicitly specify our rowBytes to be snug (no gap between rows)
230     tmp.setConfig(SkBitmap::kARGB_8888_Config, w, h, w * 4);
231     if (!tmp.allocPixels()) {
232         return false;
233     }
234 
235     tmp.lockPixels();
236 
237     bool read = fContext->readRenderTargetPixels(fRenderTarget,
238                                                  bounds.fLeft, bounds.fTop,
239                                                  bounds.width(), bounds.height(),
240                                                  kRGBA_8888_GrPixelConfig,
241                                                  tmp.getPixels());
242     tmp.unlockPixels();
243     if (!read) {
244         return false;
245     }
246 
247     tmp.swap(*bitmap);
248     return true;
249 }
250 
writePixels(const SkBitmap & bitmap,int x,int y)251 void SkGpuDevice::writePixels(const SkBitmap& bitmap, int x, int y) {
252     SkAutoLockPixels alp(bitmap);
253     if (!bitmap.readyToDraw()) {
254         return;
255     }
256     GrPixelConfig config = SkGr::BitmapConfig2PixelConfig(bitmap.config(),
257                                                           bitmap.isOpaque());
258     fContext->setRenderTarget(fRenderTarget);
259     // we aren't setting the clip or matrix, so mark as dirty
260     // we don't need to set them for this call and don't have them anyway
261     fNeedPrepareRenderTarget = true;
262 
263     fContext->writePixels(x, y, bitmap.width(), bitmap.height(),
264                           config, bitmap.getPixels(), bitmap.rowBytes());
265 }
266 
267 ///////////////////////////////////////////////////////////////////////////////
268 
convert_matrixclip(GrContext * context,const SkMatrix & matrix,const SkClipStack & clipStack,const SkRegion & clipRegion,const SkIPoint & origin)269 static void convert_matrixclip(GrContext* context, const SkMatrix& matrix,
270                                const SkClipStack& clipStack,
271                                const SkRegion& clipRegion,
272                                const SkIPoint& origin) {
273     context->setMatrix(matrix);
274 
275     SkGrClipIterator iter;
276     iter.reset(clipStack);
277     const SkIRect& skBounds = clipRegion.getBounds();
278     GrRect bounds;
279     bounds.setLTRB(GrIntToScalar(skBounds.fLeft),
280                    GrIntToScalar(skBounds.fTop),
281                    GrIntToScalar(skBounds.fRight),
282                    GrIntToScalar(skBounds.fBottom));
283     GrClip grc(&iter, GrIntToScalar(-origin.x()), GrIntToScalar(-origin.y()),
284                &bounds);
285     context->setClip(grc);
286 }
287 
288 // call this ever each draw call, to ensure that the context reflects our state,
289 // and not the state from some other canvas/device
prepareRenderTarget(const SkDraw & draw)290 void SkGpuDevice::prepareRenderTarget(const SkDraw& draw) {
291     if (fNeedPrepareRenderTarget ||
292         fContext->getRenderTarget() != fRenderTarget) {
293 
294         fContext->setRenderTarget(fRenderTarget);
295         SkASSERT(draw.fClipStack);
296         convert_matrixclip(fContext, *draw.fMatrix,
297                            *draw.fClipStack, *draw.fClip, this->getOrigin());
298         fNeedPrepareRenderTarget = false;
299     }
300 }
301 
setMatrixClip(const SkMatrix & matrix,const SkRegion & clip,const SkClipStack & clipStack)302 void SkGpuDevice::setMatrixClip(const SkMatrix& matrix, const SkRegion& clip,
303                                 const SkClipStack& clipStack) {
304     this->INHERITED::setMatrixClip(matrix, clip, clipStack);
305     // We don't need to set them now because the context may not reflect this device.
306     fNeedPrepareRenderTarget = true;
307 }
308 
gainFocus(SkCanvas * canvas,const SkMatrix & matrix,const SkRegion & clip,const SkClipStack & clipStack)309 void SkGpuDevice::gainFocus(SkCanvas* canvas, const SkMatrix& matrix,
310                             const SkRegion& clip, const SkClipStack& clipStack) {
311 
312     fContext->setRenderTarget(fRenderTarget);
313 
314     this->INHERITED::gainFocus(canvas, matrix, clip, clipStack);
315 
316     convert_matrixclip(fContext, matrix, clipStack, clip, this->getOrigin());
317 
318     if (fNeedClear) {
319         fContext->clear(NULL, 0x0);
320         fNeedClear = false;
321     }
322 }
323 
bindDeviceAsTexture(GrPaint * paint)324 bool SkGpuDevice::bindDeviceAsTexture(GrPaint* paint) {
325     if (NULL != fTexture) {
326         paint->setTexture(kBitmapTextureIdx, fTexture);
327         return true;
328     }
329     return false;
330 }
331 
332 ///////////////////////////////////////////////////////////////////////////////
333 
334 SK_COMPILE_ASSERT(SkShader::kNone_BitmapType == 0, shader_type_mismatch);
335 SK_COMPILE_ASSERT(SkShader::kDefault_BitmapType == 1, shader_type_mismatch);
336 SK_COMPILE_ASSERT(SkShader::kRadial_BitmapType == 2, shader_type_mismatch);
337 SK_COMPILE_ASSERT(SkShader::kSweep_BitmapType == 3, shader_type_mismatch);
338 SK_COMPILE_ASSERT(SkShader::kTwoPointRadial_BitmapType == 4,
339                   shader_type_mismatch);
340 SK_COMPILE_ASSERT(SkShader::kLast_BitmapType == 4, shader_type_mismatch);
341 
342 static const GrSamplerState::SampleMode sk_bmp_type_to_sample_mode[] = {
343     (GrSamplerState::SampleMode) -1,                    // kNone_BitmapType
344     GrSamplerState::kNormal_SampleMode,                 // kDefault_BitmapType
345     GrSamplerState::kRadial_SampleMode,                 // kRadial_BitmapType
346     GrSamplerState::kSweep_SampleMode,                  // kSweep_BitmapType
347     GrSamplerState::kRadial2_SampleMode,                // kTwoPointRadial_BitmapType
348 };
349 
skPaint2GrPaintNoShader(const SkPaint & skPaint,bool justAlpha,GrPaint * grPaint,bool constantColor)350 bool SkGpuDevice::skPaint2GrPaintNoShader(const SkPaint& skPaint,
351                                           bool justAlpha,
352                                           GrPaint* grPaint,
353                                           bool constantColor) {
354 
355     grPaint->fDither    = skPaint.isDither();
356     grPaint->fAntiAlias = skPaint.isAntiAlias();
357 
358     SkXfermode::Coeff sm = SkXfermode::kOne_Coeff;
359     SkXfermode::Coeff dm = SkXfermode::kISA_Coeff;
360 
361     SkXfermode* mode = skPaint.getXfermode();
362     if (mode) {
363         if (!mode->asCoeff(&sm, &dm)) {
364             SkDEBUGCODE(SkDebugf("Unsupported xfer mode.\n");)
365 #if 0
366             return false;
367 #endif
368         }
369     }
370     grPaint->fSrcBlendCoeff = sk_blend_to_grblend(sm);
371     grPaint->fDstBlendCoeff = sk_blend_to_grblend(dm);
372 
373     if (justAlpha) {
374         uint8_t alpha = skPaint.getAlpha();
375         grPaint->fColor = GrColorPackRGBA(alpha, alpha, alpha, alpha);
376         // justAlpha is currently set to true only if there is a texture,
377         // so constantColor should not also be true.
378         GrAssert(!constantColor);
379     } else {
380         grPaint->fColor = SkGr::SkColor2GrColor(skPaint.getColor());
381         grPaint->setTexture(kShaderTextureIdx, NULL);
382     }
383     SkColorFilter* colorFilter = skPaint.getColorFilter();
384     SkColor color;
385     SkXfermode::Mode filterMode;
386     if (colorFilter != NULL && colorFilter->asColorMode(&color, &filterMode)) {
387         if (!constantColor) {
388             grPaint->fColorFilterColor = SkGr::SkColor2GrColor(color);
389             grPaint->fColorFilterXfermode = filterMode;
390             return true;
391         }
392         SkColor filtered = colorFilter->filterColor(skPaint.getColor());
393         grPaint->fColor = SkGr::SkColor2GrColor(filtered);
394     }
395     grPaint->resetColorFilter();
396     return true;
397 }
398 
skPaint2GrPaintShader(const SkPaint & skPaint,SkAutoCachedTexture * act,const SkMatrix & ctm,GrPaint * grPaint,bool constantColor)399 bool SkGpuDevice::skPaint2GrPaintShader(const SkPaint& skPaint,
400                                         SkAutoCachedTexture* act,
401                                         const SkMatrix& ctm,
402                                         GrPaint* grPaint,
403                                         bool constantColor) {
404 
405     SkASSERT(NULL != act);
406 
407     SkShader* shader = skPaint.getShader();
408     if (NULL == shader) {
409         return this->skPaint2GrPaintNoShader(skPaint,
410                                              false,
411                                              grPaint,
412                                              constantColor);
413     } else if (!this->skPaint2GrPaintNoShader(skPaint, true, grPaint, false)) {
414         return false;
415     }
416 
417     SkPaint noAlphaPaint(skPaint);
418     noAlphaPaint.setAlpha(255);
419     shader->setContext(this->accessBitmap(false), noAlphaPaint, ctm);
420 
421     SkBitmap bitmap;
422     SkMatrix matrix;
423     SkShader::TileMode tileModes[2];
424     SkScalar twoPointParams[3];
425     SkShader::BitmapType bmptype = shader->asABitmap(&bitmap, &matrix,
426                                                      tileModes, twoPointParams);
427 
428     GrSamplerState::SampleMode sampleMode = sk_bmp_type_to_sample_mode[bmptype];
429     if (-1 == sampleMode) {
430         SkDebugf("shader->asABitmap() == kNone_BitmapType\n");
431         return false;
432     }
433     GrSamplerState* sampler = grPaint->getTextureSampler(kShaderTextureIdx);
434     sampler->setSampleMode(sampleMode);
435     if (skPaint.isFilterBitmap()) {
436         sampler->setFilter(GrSamplerState::kBilinear_Filter);
437     } else {
438         sampler->setFilter(GrSamplerState::kNearest_Filter);
439     }
440     sampler->setWrapX(sk_tile_mode_to_grwrap(tileModes[0]));
441     sampler->setWrapY(sk_tile_mode_to_grwrap(tileModes[1]));
442     if (GrSamplerState::kRadial2_SampleMode == sampleMode) {
443         sampler->setRadial2Params(twoPointParams[0],
444                                   twoPointParams[1],
445                                   twoPointParams[2] < 0);
446     }
447 
448     GrTexture* texture = act->set(this, bitmap, *sampler);
449     if (NULL == texture) {
450         SkDebugf("Couldn't convert bitmap to texture.\n");
451         return false;
452     }
453     grPaint->setTexture(kShaderTextureIdx, texture);
454 
455     // since our texture coords will be in local space, we wack the texture
456     // matrix to map them back into 0...1 before we load it
457     SkMatrix localM;
458     if (shader->getLocalMatrix(&localM)) {
459         SkMatrix inverse;
460         if (localM.invert(&inverse)) {
461             matrix.preConcat(inverse);
462         }
463     }
464     if (SkShader::kDefault_BitmapType == bmptype) {
465         GrScalar sx = GrFixedToScalar(GR_Fixed1 / bitmap.width());
466         GrScalar sy = GrFixedToScalar(GR_Fixed1 / bitmap.height());
467         matrix.postScale(sx, sy);
468     } else if (SkShader::kRadial_BitmapType == bmptype) {
469         GrScalar s = GrFixedToScalar(GR_Fixed1 / bitmap.width());
470         matrix.postScale(s, s);
471     }
472     sampler->setMatrix(matrix);
473 
474     return true;
475 }
476 
477 ///////////////////////////////////////////////////////////////////////////////
478 
479 class SkPositionSource {
480 public:
SkPositionSource(const SkPoint * points,int count)481     SkPositionSource(const SkPoint* points, int count)
482         : fPoints(points), fCount(count) {}
483 
count() const484     int count() const { return fCount; }
485 
writeValue(int i,GrPoint * dstPosition) const486     void writeValue(int i, GrPoint* dstPosition) const {
487         SkASSERT(i < fCount);
488         dstPosition->fX = SkScalarToGrScalar(fPoints[i].fX);
489         dstPosition->fY = SkScalarToGrScalar(fPoints[i].fY);
490     }
491 private:
492     const SkPoint*  fPoints;
493     int             fCount;
494 };
495 
496 class SkTexCoordSource {
497 public:
SkTexCoordSource(const SkPoint * coords)498     SkTexCoordSource(const SkPoint* coords)
499         : fCoords(coords) {}
500 
writeValue(int i,GrPoint * dstCoord) const501     void writeValue(int i, GrPoint* dstCoord) const {
502         dstCoord->fX = SkScalarToGrScalar(fCoords[i].fX);
503         dstCoord->fY = SkScalarToGrScalar(fCoords[i].fY);
504     }
505 private:
506     const SkPoint*  fCoords;
507 };
508 
509 class SkColorSource {
510 public:
SkColorSource(const SkColor * colors)511     SkColorSource(const SkColor* colors) : fColors(colors) {}
512 
writeValue(int i,GrColor * dstColor) const513     void writeValue(int i, GrColor* dstColor) const {
514         *dstColor = SkGr::SkColor2GrColor(fColors[i]);
515     }
516 private:
517     const SkColor* fColors;
518 };
519 
520 class SkIndexSource {
521 public:
SkIndexSource(const uint16_t * indices,int count)522     SkIndexSource(const uint16_t* indices, int count)
523         : fIndices(indices), fCount(count) {
524     }
525 
count() const526     int count() const { return fCount; }
527 
writeValue(int i,uint16_t * dstIndex) const528     void writeValue(int i, uint16_t* dstIndex) const {
529         *dstIndex = fIndices[i];
530     }
531 
532 private:
533     const uint16_t* fIndices;
534     int             fCount;
535 };
536 
537 ///////////////////////////////////////////////////////////////////////////////
538 
539 #if 0 // not currently being used so don't compile,
540 
541 // can be used for positions or texture coordinates
542 
543 class SkRectFanSource {
544 public:
545     SkRectFanSource(const SkRect& rect) : fRect(rect) {}
546 
547     int count() const { return 4; }
548 
549     void writeValue(int i, GrPoint* dstPoint) const {
550         SkASSERT(i < 4);
551         dstPoint->fX = SkScalarToGrScalar((i % 3) ? fRect.fRight :
552                                                     fRect.fLeft);
553         dstPoint->fY = SkScalarToGrScalar((i < 2) ? fRect.fTop  :
554                                                     fRect.fBottom);
555     }
556 private:
557     const SkRect&   fRect;
558 };
559 
560 class SkIRectFanSource {
561 public:
562     SkIRectFanSource(const SkIRect& rect) : fRect(rect) {}
563 
564     int count() const { return 4; }
565 
566     void writeValue(int i, GrPoint* dstPoint) const {
567         SkASSERT(i < 4);
568         dstPoint->fX = (i % 3) ? GrIntToScalar(fRect.fRight) :
569                                  GrIntToScalar(fRect.fLeft);
570         dstPoint->fY = (i < 2) ? GrIntToScalar(fRect.fTop)  :
571                                  GrIntToScalar(fRect.fBottom);
572     }
573 private:
574     const SkIRect&   fRect;
575 };
576 
577 class SkMatRectFanSource {
578 public:
579     SkMatRectFanSource(const SkRect& rect, const SkMatrix& matrix)
580         : fRect(rect), fMatrix(matrix) {}
581 
582     int count() const { return 4; }
583 
584     void writeValue(int i, GrPoint* dstPoint) const {
585         SkASSERT(i < 4);
586 
587 #if SK_SCALAR_IS_GR_SCALAR
588         fMatrix.mapXY((i % 3) ? fRect.fRight : fRect.fLeft,
589                       (i < 2) ? fRect.fTop   : fRect.fBottom,
590                       (SkPoint*)dstPoint);
591 #else
592         SkPoint dst;
593         fMatrix.mapXY((i % 3) ? fRect.fRight : fRect.fLeft,
594                       (i < 2) ? fRect.fTop   : fRect.fBottom,
595                       &dst);
596         dstPoint->fX = SkScalarToGrScalar(dst.fX);
597         dstPoint->fY = SkScalarToGrScalar(dst.fY);
598 #endif
599     }
600 private:
601     const SkRect&   fRect;
602     const SkMatrix& fMatrix;
603 };
604 
605 #endif
606 
607 ///////////////////////////////////////////////////////////////////////////////
608 
clear(SkColor color)609 void SkGpuDevice::clear(SkColor color) {
610     fContext->clear(NULL, color);
611 }
612 
drawPaint(const SkDraw & draw,const SkPaint & paint)613 void SkGpuDevice::drawPaint(const SkDraw& draw, const SkPaint& paint) {
614     CHECK_SHOULD_DRAW(draw);
615 
616     GrPaint grPaint;
617     SkAutoCachedTexture act;
618     if (!this->skPaint2GrPaintShader(paint,
619                                      &act,
620                                      *draw.fMatrix,
621                                      &grPaint,
622                                      true)) {
623         return;
624     }
625 
626     fContext->drawPaint(grPaint);
627 }
628 
629 // must be in SkCanvas::PointMode order
630 static const GrPrimitiveType gPointMode2PrimtiveType[] = {
631     kPoints_PrimitiveType,
632     kLines_PrimitiveType,
633     kLineStrip_PrimitiveType
634 };
635 
drawPoints(const SkDraw & draw,SkCanvas::PointMode mode,size_t count,const SkPoint pts[],const SkPaint & paint)636 void SkGpuDevice::drawPoints(const SkDraw& draw, SkCanvas::PointMode mode,
637                              size_t count, const SkPoint pts[], const SkPaint& paint) {
638     CHECK_SHOULD_DRAW(draw);
639 
640     SkScalar width = paint.getStrokeWidth();
641     if (width < 0) {
642         return;
643     }
644 
645     // we only handle hairlines here, else we let the SkDraw call our drawPath()
646     if (width > 0) {
647         draw.drawPoints(mode, count, pts, paint, true);
648         return;
649     }
650 
651     GrPaint grPaint;
652     SkAutoCachedTexture act;
653     if (!this->skPaint2GrPaintShader(paint,
654                                      &act,
655                                      *draw.fMatrix,
656                                      &grPaint,
657                                      true)) {
658         return;
659     }
660 
661 #if SK_SCALAR_IS_GR_SCALAR
662     fContext->drawVertices(grPaint,
663                            gPointMode2PrimtiveType[mode],
664                            count,
665                            (GrPoint*)pts,
666                            NULL,
667                            NULL,
668                            NULL,
669                            0);
670 #else
671     fContext->drawCustomVertices(grPaint,
672                                  gPointMode2PrimtiveType[mode],
673                                  SkPositionSource(pts, count));
674 #endif
675 }
676 
677 ///////////////////////////////////////////////////////////////////////////////
678 
drawRect(const SkDraw & draw,const SkRect & rect,const SkPaint & paint)679 void SkGpuDevice::drawRect(const SkDraw& draw, const SkRect& rect,
680                           const SkPaint& paint) {
681     CHECK_SHOULD_DRAW(draw);
682 
683     bool doStroke = paint.getStyle() == SkPaint::kStroke_Style;
684     SkScalar width = paint.getStrokeWidth();
685 
686     /*
687         We have special code for hairline strokes, miter-strokes, and fills.
688         Anything else we just call our path code.
689      */
690     bool usePath = doStroke && width > 0 &&
691                     paint.getStrokeJoin() != SkPaint::kMiter_Join;
692     // another reason we might need to call drawPath...
693     if (paint.getMaskFilter()) {
694         usePath = true;
695     }
696     // until we aa rotated rects...
697     if (!usePath && paint.isAntiAlias() && !draw.fMatrix->rectStaysRect()) {
698         usePath = true;
699     }
700 
701     if (usePath) {
702         SkPath path;
703         path.addRect(rect);
704         this->drawPath(draw, path, paint, NULL, true);
705         return;
706     }
707 
708     GrPaint grPaint;
709     SkAutoCachedTexture act;
710     if (!this->skPaint2GrPaintShader(paint,
711                                      &act,
712                                      *draw.fMatrix,
713                                      &grPaint,
714                                      true)) {
715         return;
716     }
717     fContext->drawRect(grPaint, rect, doStroke ? width : -1);
718 }
719 
720 #include "SkMaskFilter.h"
721 #include "SkBounder.h"
722 
drawWithMaskFilter(GrContext * context,const SkPath & path,SkMaskFilter * filter,const SkMatrix & matrix,const SkRegion & clip,SkBounder * bounder,GrPaint * grp)723 static bool drawWithMaskFilter(GrContext* context, const SkPath& path,
724                                SkMaskFilter* filter, const SkMatrix& matrix,
725                                const SkRegion& clip, SkBounder* bounder,
726                                GrPaint* grp) {
727     SkMask  srcM, dstM;
728 
729     if (!SkDraw::DrawToMask(path, &clip.getBounds(), filter, &matrix, &srcM,
730                             SkMask::kComputeBoundsAndRenderImage_CreateMode)) {
731         return false;
732     }
733 
734     SkAutoMaskImage autoSrc(&srcM, false);
735 
736     if (!filter->filterMask(&dstM, srcM, matrix, NULL)) {
737         return false;
738     }
739     // this will free-up dstM when we're done (allocated in filterMask())
740     SkAutoMaskImage autoDst(&dstM, false);
741 
742     if (clip.quickReject(dstM.fBounds)) {
743         return false;
744     }
745     if (bounder && !bounder->doIRect(dstM.fBounds)) {
746         return false;
747     }
748 
749     // we now have a device-aligned 8bit mask in dstM, ready to be drawn using
750     // the current clip (and identity matrix) and grpaint settings
751 
752     // used to compute inverse view, if necessary
753     GrMatrix ivm = context->getMatrix();
754 
755     GrAutoMatrix avm(context, GrMatrix::I());
756 
757     const GrTextureDesc desc = {
758         kNone_GrTextureFlags,
759         kNone_GrAALevel,
760         dstM.fBounds.width(),
761         dstM.fBounds.height(),
762         kAlpha_8_GrPixelConfig
763     };
764 
765     GrTexture* texture = context->createUncachedTexture(desc, dstM.fImage,
766                                                         dstM.fRowBytes);
767     if (NULL == texture) {
768         return false;
769     }
770 
771     if (grp->hasTextureOrMask() && ivm.invert(&ivm)) {
772         grp->preConcatActiveSamplerMatrices(ivm);
773     }
774 
775     static const int MASK_IDX = GrPaint::kMaxMasks - 1;
776     // we assume the last mask index is available for use
777     GrAssert(NULL == grp->getMask(MASK_IDX));
778     grp->setMask(MASK_IDX, texture);
779     texture->unref();
780     grp->getMaskSampler(MASK_IDX)->setClampNoFilter();
781 
782     GrRect d;
783     d.setLTRB(GrIntToScalar(dstM.fBounds.fLeft),
784               GrIntToScalar(dstM.fBounds.fTop),
785               GrIntToScalar(dstM.fBounds.fRight),
786               GrIntToScalar(dstM.fBounds.fBottom));
787 
788     GrMatrix m;
789     m.setTranslate(-dstM.fBounds.fLeft, -dstM.fBounds.fTop);
790     m.postIDiv(dstM.fBounds.width(), dstM.fBounds.height());
791     grp->getMaskSampler(MASK_IDX)->setMatrix(m);
792 
793     context->drawRect(*grp, d);
794     return true;
795 }
796 
drawPath(const SkDraw & draw,const SkPath & origSrcPath,const SkPaint & paint,const SkMatrix * prePathMatrix,bool pathIsMutable)797 void SkGpuDevice::drawPath(const SkDraw& draw, const SkPath& origSrcPath,
798                            const SkPaint& paint, const SkMatrix* prePathMatrix,
799                            bool pathIsMutable) {
800     CHECK_SHOULD_DRAW(draw);
801 
802     GrPaint grPaint;
803     SkAutoCachedTexture act;
804     if (!this->skPaint2GrPaintShader(paint,
805                                      &act,
806                                      *draw.fMatrix,
807                                      &grPaint,
808                                      true)) {
809         return;
810     }
811 
812     // BEGIN lift from SkDraw::drawPath()
813 
814     SkPath*         pathPtr = const_cast<SkPath*>(&origSrcPath);
815     bool            doFill = true;
816     SkPath          tmpPath;
817 
818     if (prePathMatrix) {
819         SkPath* result = pathPtr;
820 
821         if (!pathIsMutable) {
822             result = &tmpPath;
823             pathIsMutable = true;
824         }
825         // should I push prePathMatrix on our MV stack temporarily, instead
826         // of applying it here? See SkDraw.cpp
827         pathPtr->transform(*prePathMatrix, result);
828         pathPtr = result;
829     }
830     // at this point we're done with prePathMatrix
831     SkDEBUGCODE(prePathMatrix = (const SkMatrix*)0x50FF8001;)
832 
833     // This "if" is not part of the SkDraw::drawPath() lift.
834     // When we get a 1.0 wide stroke we hairline stroke it instead of creating
835     // a new stroked-path. This is motivated by canvas2D sites that draw
836     // lines as 1.0 wide stroked paths. We can consider doing an alpha-modulated-
837     // hairline for width < 1.0 when AA is enabled.
838     static const int gMatrixMask = ~(SkMatrix::kIdentity_Mask |
839                                      SkMatrix::kTranslate_Mask);
840     if (!paint.getPathEffect() &&
841         SkPaint::kStroke_Style == paint.getStyle() &&
842         !(draw.fMatrix->getType() & gMatrixMask) &&
843         SK_Scalar1 == paint.getStrokeWidth()) {
844         doFill = false;
845     }
846 
847     if (doFill && (paint.getPathEffect() ||
848                    paint.getStyle() != SkPaint::kFill_Style)) {
849         doFill = paint.getFillPath(*pathPtr, &tmpPath);
850         pathPtr = &tmpPath;
851     }
852 
853     // END lift from SkDraw::drawPath()
854 
855     if (paint.getMaskFilter()) {
856         // avoid possibly allocating a new path in transform if we can
857         SkPath* devPathPtr = pathIsMutable ? pathPtr : &tmpPath;
858 
859         // transform the path into device space
860         pathPtr->transform(*draw.fMatrix, devPathPtr);
861 
862         drawWithMaskFilter(fContext, *devPathPtr, paint.getMaskFilter(),
863                            *draw.fMatrix, *draw.fClip, draw.fBounder, &grPaint);
864         return;
865     }
866 
867     GrPathFill fill = kHairLine_PathFill;
868 
869     if (doFill) {
870         switch (pathPtr->getFillType()) {
871             case SkPath::kWinding_FillType:
872                 fill = kWinding_PathFill;
873                 break;
874             case SkPath::kEvenOdd_FillType:
875                 fill = kEvenOdd_PathFill;
876                 break;
877             case SkPath::kInverseWinding_FillType:
878                 fill = kInverseWinding_PathFill;
879                 break;
880             case SkPath::kInverseEvenOdd_FillType:
881                 fill = kInverseEvenOdd_PathFill;
882                 break;
883             default:
884                 SkDebugf("Unsupported path fill type\n");
885                 return;
886         }
887     }
888 
889     fContext->drawPath(grPaint, *pathPtr, fill);
890 }
891 
drawBitmap(const SkDraw & draw,const SkBitmap & bitmap,const SkIRect * srcRectPtr,const SkMatrix & m,const SkPaint & paint)892 void SkGpuDevice::drawBitmap(const SkDraw& draw,
893                              const SkBitmap& bitmap,
894                              const SkIRect* srcRectPtr,
895                              const SkMatrix& m,
896                              const SkPaint& paint) {
897     CHECK_SHOULD_DRAW(draw);
898 
899     SkIRect srcRect;
900     if (NULL == srcRectPtr) {
901         srcRect.set(0, 0, bitmap.width(), bitmap.height());
902     } else {
903         srcRect = *srcRectPtr;
904     }
905 
906     GrPaint grPaint;
907     if (!this->skPaint2GrPaintNoShader(paint, true, &grPaint, false)) {
908         return;
909     }
910     GrSamplerState* sampler = grPaint.getTextureSampler(kBitmapTextureIdx);
911     if (paint.isFilterBitmap()) {
912         sampler->setFilter(GrSamplerState::kBilinear_Filter);
913     } else {
914         sampler->setFilter(GrSamplerState::kNearest_Filter);
915     }
916 
917     const int maxTextureDim = fContext->getMaxTextureDimension();
918     if (bitmap.getTexture() || (bitmap.width() <= maxTextureDim &&
919                                 bitmap.height() <= maxTextureDim)) {
920         // take the fast case
921         this->internalDrawBitmap(draw, bitmap, srcRect, m, &grPaint);
922         return;
923     }
924 
925     // undo the translate done by SkCanvas
926     int DX = SkMax32(0, srcRect.fLeft);
927     int DY = SkMax32(0, srcRect.fTop);
928     // compute clip bounds in local coordinates
929     SkIRect clipRect;
930     {
931         SkRect r;
932         r.set(draw.fClip->getBounds());
933         SkMatrix matrix, inverse;
934         matrix.setConcat(*draw.fMatrix, m);
935         if (!matrix.invert(&inverse)) {
936             return;
937         }
938         inverse.mapRect(&r);
939         r.roundOut(&clipRect);
940         // apply the canvas' translate to our local clip
941         clipRect.offset(DX, DY);
942     }
943 
944     int nx = bitmap.width() / maxTextureDim;
945     int ny = bitmap.height() / maxTextureDim;
946     for (int x = 0; x <= nx; x++) {
947         for (int y = 0; y <= ny; y++) {
948             SkIRect tileR;
949             tileR.set(x * maxTextureDim, y * maxTextureDim,
950                       (x + 1) * maxTextureDim, (y + 1) * maxTextureDim);
951             if (!SkIRect::Intersects(tileR, clipRect)) {
952                 continue;
953             }
954 
955             SkIRect srcR = tileR;
956             if (!srcR.intersect(srcRect)) {
957                 continue;
958             }
959 
960             SkBitmap tmpB;
961             if (bitmap.extractSubset(&tmpB, tileR)) {
962                 // now offset it to make it "local" to our tmp bitmap
963                 srcR.offset(-tileR.fLeft, -tileR.fTop);
964 
965                 SkMatrix tmpM(m);
966                 {
967                     int dx = tileR.fLeft - DX + SkMax32(0, srcR.fLeft);
968                     int dy = tileR.fTop -  DY + SkMax32(0, srcR.fTop);
969                     tmpM.preTranslate(SkIntToScalar(dx), SkIntToScalar(dy));
970                 }
971                 this->internalDrawBitmap(draw, tmpB, srcR, tmpM, &grPaint);
972             }
973         }
974     }
975 }
976 
977 /*
978  *  This is called by drawBitmap(), which has to handle images that may be too
979  *  large to be represented by a single texture.
980  *
981  *  internalDrawBitmap assumes that the specified bitmap will fit in a texture
982  *  and that non-texture portion of the GrPaint has already been setup.
983  */
internalDrawBitmap(const SkDraw & draw,const SkBitmap & bitmap,const SkIRect & srcRect,const SkMatrix & m,GrPaint * grPaint)984 void SkGpuDevice::internalDrawBitmap(const SkDraw& draw,
985                                      const SkBitmap& bitmap,
986                                      const SkIRect& srcRect,
987                                      const SkMatrix& m,
988                                      GrPaint* grPaint) {
989     SkASSERT(bitmap.width() <= fContext->getMaxTextureDimension() &&
990              bitmap.height() <= fContext->getMaxTextureDimension());
991 
992     SkAutoLockPixels alp(bitmap);
993     if (!bitmap.getTexture() && !bitmap.readyToDraw()) {
994         return;
995     }
996 
997     GrSamplerState* sampler = grPaint->getTextureSampler(kBitmapTextureIdx);
998 
999     sampler->setWrapX(GrSamplerState::kClamp_WrapMode);
1000     sampler->setWrapY(GrSamplerState::kClamp_WrapMode);
1001     sampler->setSampleMode(GrSamplerState::kNormal_SampleMode);
1002     sampler->setMatrix(GrMatrix::I());
1003 
1004     GrTexture* texture;
1005     SkAutoCachedTexture act(this, bitmap, *sampler, &texture);
1006     if (NULL == texture) {
1007         return;
1008     }
1009 
1010     grPaint->setTexture(kShaderTextureIdx, texture);
1011 
1012     GrRect dstRect = SkRect::MakeWH(GrIntToScalar(srcRect.width()),
1013                                     GrIntToScalar(srcRect.height()));
1014     GrRect paintRect;
1015     paintRect.setLTRB(GrFixedToScalar((srcRect.fLeft << 16) / bitmap.width()),
1016                       GrFixedToScalar((srcRect.fTop << 16) / bitmap.height()),
1017                       GrFixedToScalar((srcRect.fRight << 16) / bitmap.width()),
1018                       GrFixedToScalar((srcRect.fBottom << 16) / bitmap.height()));
1019 
1020     if (GrSamplerState::kNearest_Filter != sampler->getFilter() &&
1021         (srcRect.width() < bitmap.width() ||
1022          srcRect.height() < bitmap.height())) {
1023         // If drawing a subrect of the bitmap and filtering is enabled,
1024         // use a constrained texture domain to avoid color bleeding
1025         GrScalar left, top, right, bottom;
1026         if (srcRect.width() > 1) {
1027             GrScalar border = GR_ScalarHalf / bitmap.width();
1028             left = paintRect.left() + border;
1029             right = paintRect.right() - border;
1030         } else {
1031             left = right = GrScalarHalf(paintRect.left() + paintRect.right());
1032         }
1033         if (srcRect.height() > 1) {
1034             GrScalar border = GR_ScalarHalf / bitmap.height();
1035             top = paintRect.top() + border;
1036             bottom = paintRect.bottom() - border;
1037         } else {
1038             top = bottom = GrScalarHalf(paintRect.top() + paintRect.bottom());
1039         }
1040         GrRect textureDomain;
1041         textureDomain.setLTRB(left, top, right, bottom);
1042         sampler->setTextureDomain(textureDomain);
1043     }
1044 
1045     fContext->drawRectToRect(*grPaint, dstRect, paintRect, &m);
1046 }
1047 
drawSprite(const SkDraw & draw,const SkBitmap & bitmap,int left,int top,const SkPaint & paint)1048 void SkGpuDevice::drawSprite(const SkDraw& draw, const SkBitmap& bitmap,
1049                             int left, int top, const SkPaint& paint) {
1050     CHECK_SHOULD_DRAW(draw);
1051 
1052     SkAutoLockPixels alp(bitmap);
1053     if (!bitmap.getTexture() && !bitmap.readyToDraw()) {
1054         return;
1055     }
1056 
1057     GrPaint grPaint;
1058     if(!this->skPaint2GrPaintNoShader(paint, true, &grPaint, false)) {
1059         return;
1060     }
1061 
1062     GrAutoMatrix avm(fContext, GrMatrix::I());
1063 
1064     GrSamplerState* sampler = grPaint.getTextureSampler(kBitmapTextureIdx);
1065 
1066     GrTexture* texture;
1067     sampler->setClampNoFilter();
1068     SkAutoCachedTexture act(this, bitmap, *sampler, &texture);
1069 
1070     grPaint.setTexture(kBitmapTextureIdx, texture);
1071 
1072     fContext->drawRectToRect(grPaint,
1073                              GrRect::MakeXYWH(GrIntToScalar(left),
1074                                               GrIntToScalar(top),
1075                                               GrIntToScalar(bitmap.width()),
1076                                               GrIntToScalar(bitmap.height())),
1077                              GrRect::MakeWH(GR_Scalar1, GR_Scalar1));
1078 }
1079 
drawDevice(const SkDraw & draw,SkDevice * dev,int x,int y,const SkPaint & paint)1080 void SkGpuDevice::drawDevice(const SkDraw& draw, SkDevice* dev,
1081                             int x, int y, const SkPaint& paint) {
1082     CHECK_SHOULD_DRAW(draw);
1083 
1084     GrPaint grPaint;
1085     if (!((SkGpuDevice*)dev)->bindDeviceAsTexture(&grPaint) ||
1086         !this->skPaint2GrPaintNoShader(paint, true, &grPaint, false)) {
1087         return;
1088     }
1089 
1090     SkASSERT(NULL != grPaint.getTexture(0));
1091 
1092     const SkBitmap& bm = dev->accessBitmap(false);
1093     int w = bm.width();
1094     int h = bm.height();
1095 
1096     GrAutoMatrix avm(fContext, GrMatrix::I());
1097 
1098     grPaint.getTextureSampler(kBitmapTextureIdx)->setClampNoFilter();
1099 
1100     fContext->drawRectToRect(grPaint,
1101                              GrRect::MakeXYWH(GrIntToScalar(x),
1102                                               GrIntToScalar(y),
1103                                               GrIntToScalar(w),
1104                                               GrIntToScalar(h)),
1105                              GrRect::MakeWH(GR_Scalar1, GR_Scalar1));
1106 }
1107 
1108 ///////////////////////////////////////////////////////////////////////////////
1109 
1110 // must be in SkCanvas::VertexMode order
1111 static const GrPrimitiveType gVertexMode2PrimitiveType[] = {
1112     kTriangles_PrimitiveType,
1113     kTriangleStrip_PrimitiveType,
1114     kTriangleFan_PrimitiveType,
1115 };
1116 
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)1117 void SkGpuDevice::drawVertices(const SkDraw& draw, SkCanvas::VertexMode vmode,
1118                               int vertexCount, const SkPoint vertices[],
1119                               const SkPoint texs[], const SkColor colors[],
1120                               SkXfermode* xmode,
1121                               const uint16_t indices[], int indexCount,
1122                               const SkPaint& paint) {
1123     CHECK_SHOULD_DRAW(draw);
1124 
1125     GrPaint grPaint;
1126     SkAutoCachedTexture act;
1127     // we ignore the shader if texs is null.
1128     if (NULL == texs) {
1129         if (!this->skPaint2GrPaintNoShader(paint,
1130                                            false,
1131                                            &grPaint,
1132                                            NULL == colors)) {
1133             return;
1134         }
1135     } else {
1136         if (!this->skPaint2GrPaintShader(paint, &act,
1137                                          *draw.fMatrix,
1138                                          &grPaint,
1139                                          NULL == colors)) {
1140             return;
1141         }
1142     }
1143 
1144     if (NULL != xmode && NULL != texs && NULL != colors) {
1145         SkXfermode::Mode mode;
1146         if (!SkXfermode::IsMode(xmode, &mode) ||
1147             SkXfermode::kMultiply_Mode != mode) {
1148             SkDebugf("Unsupported vertex-color/texture xfer mode.\n");
1149 #if 0
1150             return
1151 #endif
1152         }
1153     }
1154 
1155 #if SK_SCALAR_IS_GR_SCALAR
1156     // even if GrColor and SkColor byte offsets match we need
1157     // to perform pre-multiply.
1158     if (NULL == colors) {
1159         fContext->drawVertices(grPaint,
1160                                gVertexMode2PrimitiveType[vmode],
1161                                vertexCount,
1162                                (GrPoint*) vertices,
1163                                (GrPoint*) texs,
1164                                NULL,
1165                                indices,
1166                                indexCount);
1167     } else
1168 #endif
1169     {
1170         SkTexCoordSource texSrc(texs);
1171         SkColorSource colSrc(colors);
1172         SkIndexSource idxSrc(indices, indexCount);
1173 
1174         fContext->drawCustomVertices(grPaint,
1175                                      gVertexMode2PrimitiveType[vmode],
1176                                      SkPositionSource(vertices, vertexCount),
1177                                      (NULL == texs) ? NULL : &texSrc,
1178                                      (NULL == colors) ? NULL : &colSrc,
1179                                      (NULL == indices) ? NULL : &idxSrc);
1180     }
1181 }
1182 
1183 ///////////////////////////////////////////////////////////////////////////////
1184 
GlyphCacheAuxProc(void * data)1185 static void GlyphCacheAuxProc(void* data) {
1186     delete (GrFontScaler*)data;
1187 }
1188 
get_gr_font_scaler(SkGlyphCache * cache)1189 static GrFontScaler* get_gr_font_scaler(SkGlyphCache* cache) {
1190     void* auxData;
1191     GrFontScaler* scaler = NULL;
1192     if (cache->getAuxProcData(GlyphCacheAuxProc, &auxData)) {
1193         scaler = (GrFontScaler*)auxData;
1194     }
1195     if (NULL == scaler) {
1196         scaler = new SkGrFontScaler(cache);
1197         cache->setAuxProc(GlyphCacheAuxProc, scaler);
1198     }
1199     return scaler;
1200 }
1201 
SkGPU_Draw1Glyph(const SkDraw1Glyph & state,SkFixed fx,SkFixed fy,const SkGlyph & glyph)1202 static void SkGPU_Draw1Glyph(const SkDraw1Glyph& state,
1203                              SkFixed fx, SkFixed fy,
1204                              const SkGlyph& glyph) {
1205     SkASSERT(glyph.fWidth > 0 && glyph.fHeight > 0);
1206 
1207     GrSkDrawProcs* procs = (GrSkDrawProcs*)state.fDraw->fProcs;
1208 
1209     if (NULL == procs->fFontScaler) {
1210         procs->fFontScaler = get_gr_font_scaler(state.fCache);
1211     }
1212 
1213     /*
1214      *  What should we do with fy? (assuming horizontal/latin text)
1215      *
1216      *  The raster code calls SkFixedFloorToFixed on it, as it does with fx.
1217      *  It calls that rather than round, because our caller has already added
1218      *  SK_FixedHalf, so that calling floor gives us the rounded integer.
1219      *
1220      *  Test code between raster and gpu (they should draw the same)
1221      *
1222      *      canvas->drawText("Hamburgefons", 12, 0, 16.5f, paint);
1223      *
1224      *  Perhaps we should only perform this integralization if there is no
1225      *  fExtMatrix...
1226      */
1227     fy = SkFixedFloorToFixed(fy);
1228 
1229     procs->fTextContext->drawPackedGlyph(GrGlyph::Pack(glyph.getGlyphID(), fx, 0),
1230                                          SkFixedFloorToFixed(fx), fy,
1231                                          procs->fFontScaler);
1232 }
1233 
initDrawForText(GrTextContext * context)1234 SkDrawProcs* SkGpuDevice::initDrawForText(GrTextContext* context) {
1235 
1236     // deferred allocation
1237     if (NULL == fDrawProcs) {
1238         fDrawProcs = new GrSkDrawProcs;
1239         fDrawProcs->fD1GProc = SkGPU_Draw1Glyph;
1240         fDrawProcs->fContext = fContext;
1241     }
1242 
1243     // init our (and GL's) state
1244     fDrawProcs->fTextContext = context;
1245     fDrawProcs->fFontScaler = NULL;
1246     return fDrawProcs;
1247 }
1248 
drawText(const SkDraw & draw,const void * text,size_t byteLength,SkScalar x,SkScalar y,const SkPaint & paint)1249 void SkGpuDevice::drawText(const SkDraw& draw, const void* text,
1250                           size_t byteLength, SkScalar x, SkScalar y,
1251                           const SkPaint& paint) {
1252     CHECK_SHOULD_DRAW(draw);
1253 
1254     if (draw.fMatrix->getType() & SkMatrix::kPerspective_Mask) {
1255         // this guy will just call our drawPath()
1256         draw.drawText((const char*)text, byteLength, x, y, paint);
1257     } else {
1258         SkDraw myDraw(draw);
1259 
1260         GrPaint grPaint;
1261         SkAutoCachedTexture act;
1262 
1263         if (!this->skPaint2GrPaintShader(paint,
1264                                          &act,
1265                                          *draw.fMatrix,
1266                                          &grPaint,
1267                                          true)) {
1268             return;
1269         }
1270         GrTextContext context(fContext, grPaint, draw.fExtMatrix);
1271         myDraw.fProcs = this->initDrawForText(&context);
1272         this->INHERITED::drawText(myDraw, text, byteLength, x, y, paint);
1273     }
1274 }
1275 
drawPosText(const SkDraw & draw,const void * text,size_t byteLength,const SkScalar pos[],SkScalar constY,int scalarsPerPos,const SkPaint & paint)1276 void SkGpuDevice::drawPosText(const SkDraw& draw, const void* text,
1277                              size_t byteLength, const SkScalar pos[],
1278                              SkScalar constY, int scalarsPerPos,
1279                              const SkPaint& paint) {
1280     CHECK_SHOULD_DRAW(draw);
1281 
1282     if (draw.fMatrix->getType() & SkMatrix::kPerspective_Mask) {
1283         // this guy will just call our drawPath()
1284         draw.drawPosText((const char*)text, byteLength, pos, constY,
1285                          scalarsPerPos, paint);
1286     } else {
1287         SkDraw myDraw(draw);
1288 
1289         GrPaint grPaint;
1290         SkAutoCachedTexture act;
1291         if (!this->skPaint2GrPaintShader(paint,
1292                                          &act,
1293                                          *draw.fMatrix,
1294                                          &grPaint,
1295                                          true)) {
1296             return;
1297         }
1298 
1299         GrTextContext context(fContext, grPaint, draw.fExtMatrix);
1300         myDraw.fProcs = this->initDrawForText(&context);
1301         this->INHERITED::drawPosText(myDraw, text, byteLength, pos, constY,
1302                                      scalarsPerPos, paint);
1303     }
1304 }
1305 
drawTextOnPath(const SkDraw & draw,const void * text,size_t len,const SkPath & path,const SkMatrix * m,const SkPaint & paint)1306 void SkGpuDevice::drawTextOnPath(const SkDraw& draw, const void* text,
1307                                 size_t len, const SkPath& path,
1308                                 const SkMatrix* m, const SkPaint& paint) {
1309     CHECK_SHOULD_DRAW(draw);
1310 
1311     SkASSERT(draw.fDevice == this);
1312     draw.drawTextOnPath((const char*)text, len, path, m, paint);
1313 }
1314 
1315 ///////////////////////////////////////////////////////////////////////////////
1316 
filterTextFlags(const SkPaint & paint,TextFlags * flags)1317 bool SkGpuDevice::filterTextFlags(const SkPaint& paint, TextFlags* flags) {
1318     if (!paint.isLCDRenderText()) {
1319         // we're cool with the paint as is
1320         return false;
1321     }
1322 
1323     if (paint.getShader() ||
1324         paint.getXfermode() || // unless its srcover
1325         paint.getMaskFilter() ||
1326         paint.getRasterizer() ||
1327         paint.getColorFilter() ||
1328         paint.getPathEffect() ||
1329         paint.isFakeBoldText() ||
1330         paint.getStyle() != SkPaint::kFill_Style) {
1331         // turn off lcd
1332         flags->fFlags = paint.getFlags() & ~SkPaint::kLCDRenderText_Flag;
1333         flags->fHinting = paint.getHinting();
1334         return true;
1335     }
1336     // we're cool with the paint as is
1337     return false;
1338 }
1339 
1340 ///////////////////////////////////////////////////////////////////////////////
1341 
lockCachedTexture(const SkBitmap & bitmap,const GrSamplerState & sampler,GrTexture ** texture,bool forDeviceRenderTarget)1342 SkGpuDevice::TexCache* SkGpuDevice::lockCachedTexture(const SkBitmap& bitmap,
1343                                                       const GrSamplerState& sampler,
1344                                                       GrTexture** texture,
1345                                                       bool forDeviceRenderTarget) {
1346     GrTexture* newTexture = NULL;
1347     GrTextureEntry* entry = NULL;
1348     GrContext* ctx = this->context();
1349 
1350     if (forDeviceRenderTarget) {
1351         const GrTextureDesc desc = {
1352             kRenderTarget_GrTextureFlagBit,
1353             kNone_GrAALevel,
1354             bitmap.width(),
1355             bitmap.height(),
1356             SkGr::Bitmap2PixelConfig(bitmap)
1357         };
1358         entry = ctx->lockKeylessTexture(desc);
1359     } else {
1360         uint32_t p0, p1;
1361         p0 = bitmap.getGenerationID();
1362         p1 = bitmap.pixelRefOffset();
1363 
1364         GrTextureKey key(p0, p1, bitmap.width(), bitmap.height());
1365         entry = ctx->findAndLockTexture(&key, sampler);
1366 
1367         if (NULL == entry) {
1368             entry = sk_gr_create_bitmap_texture(ctx, &key, sampler, bitmap);
1369             if (NULL == entry) {
1370                 GrPrintf("---- failed to create texture for cache [%d %d]\n",
1371                          bitmap.width(), bitmap.height());
1372             }
1373         }
1374     }
1375 
1376     if (NULL != entry) {
1377         newTexture = entry->texture();
1378         if (texture) {
1379             *texture = newTexture;
1380         }
1381     }
1382     return (TexCache*)entry;
1383 }
1384 
unlockCachedTexture(TexCache * cache)1385 void SkGpuDevice::unlockCachedTexture(TexCache* cache) {
1386     this->context()->unlockTexture((GrTextureEntry*)cache);
1387 }
1388 
1389 ///////////////////////////////////////////////////////////////////////////////
1390 
SkGpuDeviceFactory(GrContext * context,GrRenderTarget * rootRenderTarget)1391 SkGpuDeviceFactory::SkGpuDeviceFactory(GrContext* context,
1392                                        GrRenderTarget* rootRenderTarget) {
1393     GrAssert(NULL != context);
1394     GrAssert(NULL != rootRenderTarget);
1395 
1396     // check this now rather than passing this value to SkGpuDevice cons.
1397     // we want the rt that is bound *now* in the 3D API, not the one
1398     // at the time of newDevice.
1399     if (SkGpuDevice::Current3DApiRenderTarget() == rootRenderTarget) {
1400         fRootRenderTarget = context->createRenderTargetFrom3DApiState();
1401     } else {
1402         fRootRenderTarget = rootRenderTarget;
1403         rootRenderTarget->ref();
1404     }
1405 
1406     fContext = context;
1407     context->ref();
1408 
1409     fRootTexture = NULL;
1410 }
1411 
SkGpuDeviceFactory(GrContext * context,GrTexture * rootRenderTargetTexture)1412 SkGpuDeviceFactory::SkGpuDeviceFactory(GrContext* context, GrTexture* rootRenderTargetTexture) {
1413     GrAssert(NULL != context);
1414     GrAssert(NULL != rootRenderTargetTexture);
1415     GrAssert(NULL != rootRenderTargetTexture->asRenderTarget());
1416 
1417     fRootTexture = rootRenderTargetTexture;
1418     rootRenderTargetTexture->ref();
1419 
1420     fRootRenderTarget = rootRenderTargetTexture->asRenderTarget();
1421     fRootRenderTarget->ref();
1422 
1423     fContext = context;
1424     context->ref();
1425 }
1426 
~SkGpuDeviceFactory()1427 SkGpuDeviceFactory::~SkGpuDeviceFactory() {
1428     fContext->unref();
1429     fRootRenderTarget->unref();
1430     GrSafeUnref(fRootTexture);
1431 }
1432 
newDevice(SkCanvas *,SkBitmap::Config config,int width,int height,bool isOpaque,bool isLayer)1433 SkDevice* SkGpuDeviceFactory::newDevice(SkCanvas*, SkBitmap::Config config,
1434                                         int width, int height,
1435                                         bool isOpaque, bool isLayer) {
1436     SkBitmap bm;
1437     bm.setConfig(config, width, height);
1438     bm.setIsOpaque(isOpaque);
1439     return new SkGpuDevice(fContext, bm, isLayer ?  NULL : fRootRenderTarget);
1440 }
1441