| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright 2014 Google Inc. | 2 * Copyright 2014 Google Inc. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license that can be | 4 * Use of this source code is governed by a BSD-style license that can be |
| 5 * found in the LICENSE file. | 5 * found in the LICENSE file. |
| 6 */ | 6 */ |
| 7 | 7 |
| 8 #include "SkLayerInfo.h" | |
| 9 #include "SkRecordDraw.h" | 8 #include "SkRecordDraw.h" |
| 10 #include "SkPatchUtils.h" | 9 #include "SkPatchUtils.h" |
| 11 | 10 |
| 12 void SkRecordDraw(const SkRecord& record, | 11 void SkRecordDraw(const SkRecord& record, |
| 13 SkCanvas* canvas, | 12 SkCanvas* canvas, |
| 14 SkPicture const* const drawablePicts[], | 13 SkPicture const* const drawablePicts[], |
| 15 SkDrawable* const drawables[], | 14 SkDrawable* const drawables[], |
| 16 int drawableCount, | 15 int drawableCount, |
| 17 const SkBBoxHierarchy* bbh, | 16 const SkBBoxHierarchy* bbh, |
| 18 SkPicture::AbortCallback* callback) { | 17 SkPicture::AbortCallback* callback) { |
| (...skipping 579 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 598 // identity-space bounds of the current clip (fCurrentClipBounds). | 597 // identity-space bounds of the current clip (fCurrentClipBounds). |
| 599 int fCurrentOp; | 598 int fCurrentOp; |
| 600 SkMatrix fCTM; | 599 SkMatrix fCTM; |
| 601 Bounds fCurrentClipBounds; | 600 Bounds fCurrentClipBounds; |
| 602 | 601 |
| 603 // Used to track the bounds of Save/Restore blocks and the control ops insid
e them. | 602 // Used to track the bounds of Save/Restore blocks and the control ops insid
e them. |
| 604 SkTDArray<SaveBounds> fSaveStack; | 603 SkTDArray<SaveBounds> fSaveStack; |
| 605 SkTDArray<int> fControlIndices; | 604 SkTDArray<int> fControlIndices; |
| 606 }; | 605 }; |
| 607 | 606 |
| 608 // SkRecord visitor to gather saveLayer/restore information. | |
| 609 class CollectLayers : SkNoncopyable { | |
| 610 public: | |
| 611 CollectLayers(const SkRect& cullRect, const SkRecord& record, SkRect bounds[
], | |
| 612 const SkBigPicture::SnapshotArray* pictList, SkLayerInfo* acce
lData) | |
| 613 : fSaveLayersInStack(0) | |
| 614 , fAccelData(accelData) | |
| 615 , fPictList(pictList) | |
| 616 , fFillBounds(cullRect, record, bounds) | |
| 617 {} | |
| 618 | |
| 619 void cleanUp() { | |
| 620 // fFillBounds must perform its cleanUp first so that all the bounding | |
| 621 // boxes associated with unbalanced restores are updated (prior to | |
| 622 // fetching their bound in popSaveLayerInfo). | |
| 623 fFillBounds.cleanUp(); | |
| 624 while (!fSaveLayerStack.isEmpty()) { | |
| 625 this->popSaveLayerInfo(); | |
| 626 } | |
| 627 } | |
| 628 | |
| 629 void setCurrentOp(int currentOp) { fFillBounds.setCurrentOp(currentOp); } | |
| 630 | |
| 631 | |
| 632 template <typename T> void operator()(const T& op) { | |
| 633 fFillBounds(op); | |
| 634 this->trackSaveLayers(op); | |
| 635 } | |
| 636 | |
| 637 private: | |
| 638 struct SaveLayerInfo { | |
| 639 SaveLayerInfo() { } | |
| 640 SaveLayerInfo(int opIndex, bool isSaveLayer, const SkRect* bounds, const
SkPaint* paint) | |
| 641 : fStartIndex(opIndex) | |
| 642 , fIsSaveLayer(isSaveLayer) | |
| 643 , fHasNestedSaveLayer(false) | |
| 644 , fBounds(bounds ? *bounds : SkRect::MakeEmpty()) | |
| 645 , fPaint(paint) { | |
| 646 } | |
| 647 | |
| 648 int fStartIndex; | |
| 649 bool fIsSaveLayer; | |
| 650 bool fHasNestedSaveLayer; | |
| 651 SkRect fBounds; | |
| 652 const SkPaint* fPaint; | |
| 653 }; | |
| 654 | |
| 655 template <typename T> void trackSaveLayers(const T& op) { | |
| 656 /* most ops aren't involved in saveLayers */ | |
| 657 } | |
| 658 void trackSaveLayers(const Save& s) { this->pushSaveLayerInfo(false, nullptr
, nullptr); } | |
| 659 void trackSaveLayers(const SaveLayer& sl) { this->pushSaveLayerInfo(true, sl
.bounds, sl.paint); } | |
| 660 void trackSaveLayers(const Restore& r) { this->popSaveLayerInfo(); } | |
| 661 | |
| 662 void trackSaveLayersForPicture(const SkPicture* picture, const SkPaint* pain
t) { | |
| 663 // For sub-pictures, we wrap their layer information within the parent | |
| 664 // picture's rendering hierarchy | |
| 665 const SkLayerInfo* childData = nullptr; | |
| 666 if (const SkBigPicture* bp = picture->asSkBigPicture()) { | |
| 667 childData = static_cast<const SkLayerInfo*>(bp->accelData()); | |
| 668 } | |
| 669 if (!childData) { | |
| 670 // If the child layer hasn't been generated with saveLayer data we | |
| 671 // assume the worst (i.e., that it does contain layers which nest | |
| 672 // inside existing layers). Layers within sub-pictures that don't | |
| 673 // have saveLayer data cannot be hoisted. | |
| 674 // TODO: could the analysis data be use to fine tune this? | |
| 675 this->updateStackForSaveLayer(); | |
| 676 return; | |
| 677 } | |
| 678 | |
| 679 for (int i = 0; i < childData->numBlocks(); ++i) { | |
| 680 const SkLayerInfo::BlockInfo& src = childData->block(i); | |
| 681 | |
| 682 FillBounds::Bounds newBound = fFillBounds.adjustAndMap(src.fBounds,
paint); | |
| 683 if (newBound.isEmpty()) { | |
| 684 continue; | |
| 685 } | |
| 686 | |
| 687 this->updateStackForSaveLayer(); | |
| 688 | |
| 689 SkLayerInfo::BlockInfo& dst = fAccelData->addBlock(); | |
| 690 | |
| 691 // If src.fPicture is nullptr the layer is in dp.picture; otherwise | |
| 692 // it belongs to a sub-picture. | |
| 693 dst.fPicture = src.fPicture ? src.fPicture : picture; | |
| 694 dst.fPicture->ref(); | |
| 695 dst.fBounds = newBound; | |
| 696 dst.fSrcBounds = src.fSrcBounds; | |
| 697 dst.fLocalMat = src.fLocalMat; | |
| 698 dst.fPreMat = src.fPreMat; | |
| 699 dst.fPreMat.postConcat(fFillBounds.ctm()); | |
| 700 if (src.fPaint) { | |
| 701 dst.fPaint = new SkPaint(*src.fPaint); | |
| 702 } | |
| 703 dst.fSaveLayerOpID = src.fSaveLayerOpID; | |
| 704 dst.fRestoreOpID = src.fRestoreOpID; | |
| 705 dst.fHasNestedLayers = src.fHasNestedLayers; | |
| 706 dst.fIsNested = fSaveLayersInStack > 0 || src.fIsNested; | |
| 707 | |
| 708 // Store 'saveLayer ops from enclosing picture' + drawPict op + 'ops
from sub-picture' | |
| 709 dst.fKeySize = fSaveLayerOpStack.count() + src.fKeySize + 1; | |
| 710 dst.fKey = new int[dst.fKeySize]; | |
| 711 sk_careful_memcpy(dst.fKey, fSaveLayerOpStack.begin(), | |
| 712 fSaveLayerOpStack.count() * sizeof(int)); | |
| 713 dst.fKey[fSaveLayerOpStack.count()] = fFillBounds.currentOp(); | |
| 714 memcpy(&dst.fKey[fSaveLayerOpStack.count()+1], src.fKey, src.fKeySiz
e * sizeof(int)); | |
| 715 } | |
| 716 } | |
| 717 | |
| 718 void trackSaveLayers(const DrawPicture& dp) { | |
| 719 this->trackSaveLayersForPicture(dp.picture, dp.paint); | |
| 720 } | |
| 721 | |
| 722 void trackSaveLayers(const DrawDrawable& dp) { | |
| 723 SkASSERT(fPictList); | |
| 724 SkASSERT(dp.index >= 0 && dp.index < fPictList->count()); | |
| 725 const SkPaint* paint = nullptr; // drawables don't get a side-car pai
nt | |
| 726 this->trackSaveLayersForPicture(fPictList->begin()[dp.index], paint); | |
| 727 } | |
| 728 | |
| 729 // Inform all the saveLayers already on the stack that they now have a | |
| 730 // nested saveLayer inside them | |
| 731 void updateStackForSaveLayer() { | |
| 732 for (int index = fSaveLayerStack.count() - 1; index >= 0; --index) { | |
| 733 if (fSaveLayerStack[index].fHasNestedSaveLayer) { | |
| 734 break; | |
| 735 } | |
| 736 fSaveLayerStack[index].fHasNestedSaveLayer = true; | |
| 737 if (fSaveLayerStack[index].fIsSaveLayer) { | |
| 738 break; | |
| 739 } | |
| 740 } | |
| 741 } | |
| 742 | |
| 743 void pushSaveLayerInfo(bool isSaveLayer, const SkRect* bounds, const SkPaint
* paint) { | |
| 744 if (isSaveLayer) { | |
| 745 this->updateStackForSaveLayer(); | |
| 746 ++fSaveLayersInStack; | |
| 747 fSaveLayerOpStack.push(fFillBounds.currentOp()); | |
| 748 } | |
| 749 | |
| 750 fSaveLayerStack.push(SaveLayerInfo(fFillBounds.currentOp(), isSaveLayer,
bounds, paint)); | |
| 751 } | |
| 752 | |
| 753 void popSaveLayerInfo() { | |
| 754 if (fSaveLayerStack.count() <= 0) { | |
| 755 SkASSERT(false); | |
| 756 return; | |
| 757 } | |
| 758 | |
| 759 SkASSERT(fSaveLayersInStack == fSaveLayerOpStack.count()); | |
| 760 | |
| 761 SaveLayerInfo sli; | |
| 762 fSaveLayerStack.pop(&sli); | |
| 763 | |
| 764 if (!sli.fIsSaveLayer) { | |
| 765 return; | |
| 766 } | |
| 767 | |
| 768 --fSaveLayersInStack; | |
| 769 | |
| 770 SkLayerInfo::BlockInfo& block = fAccelData->addBlock(); | |
| 771 | |
| 772 SkASSERT(nullptr == block.fPicture); // This layer is in the top-most p
icture | |
| 773 | |
| 774 block.fBounds = fFillBounds.getBounds(sli.fStartIndex); | |
| 775 block.fLocalMat = fFillBounds.ctm(); | |
| 776 block.fPreMat = SkMatrix::I(); | |
| 777 if (sli.fPaint) { | |
| 778 block.fPaint = new SkPaint(*sli.fPaint); | |
| 779 } | |
| 780 | |
| 781 block.fSrcBounds = sli.fBounds; | |
| 782 block.fSaveLayerOpID = sli.fStartIndex; | |
| 783 block.fRestoreOpID = fFillBounds.currentOp(); | |
| 784 block.fHasNestedLayers = sli.fHasNestedSaveLayer; | |
| 785 block.fIsNested = fSaveLayersInStack > 0; | |
| 786 | |
| 787 block.fKeySize = fSaveLayerOpStack.count(); | |
| 788 block.fKey = new int[block.fKeySize]; | |
| 789 memcpy(block.fKey, fSaveLayerOpStack.begin(), block.fKeySize * sizeof(in
t)); | |
| 790 | |
| 791 fSaveLayerOpStack.pop(); | |
| 792 } | |
| 793 | |
| 794 // Used to collect saveLayer information for layer hoisting | |
| 795 int fSaveLayersInStack; | |
| 796 SkTDArray<SaveLayerInfo> fSaveLayerStack; | |
| 797 // The op code indices of all the currently active saveLayers | |
| 798 SkTDArray<int> fSaveLayerOpStack; | |
| 799 SkLayerInfo* fAccelData; | |
| 800 const SkBigPicture::SnapshotArray* fPictList; | |
| 801 | |
| 802 SkRecords::FillBounds fFillBounds; | |
| 803 }; | |
| 804 | |
| 805 } // namespace SkRecords | 607 } // namespace SkRecords |
| 806 | 608 |
| 807 void SkRecordFillBounds(const SkRect& cullRect, const SkRecord& record, SkRect b
ounds[]) { | 609 void SkRecordFillBounds(const SkRect& cullRect, const SkRecord& record, SkRect b
ounds[]) { |
| 808 SkRecords::FillBounds visitor(cullRect, record, bounds); | 610 SkRecords::FillBounds visitor(cullRect, record, bounds); |
| 809 for (int curOp = 0; curOp < record.count(); curOp++) { | 611 for (int curOp = 0; curOp < record.count(); curOp++) { |
| 810 visitor.setCurrentOp(curOp); | 612 visitor.setCurrentOp(curOp); |
| 811 record.visit(curOp, visitor); | 613 record.visit(curOp, visitor); |
| 812 } | 614 } |
| 813 visitor.cleanUp(); | 615 visitor.cleanUp(); |
| 814 } | 616 } |
| 815 | 617 |
| 816 void SkRecordComputeLayers(const SkRect& cullRect, const SkRecord& record, SkRec
t bounds[], | |
| 817 const SkBigPicture::SnapshotArray* pictList, SkLayerI
nfo* data) { | |
| 818 SkRecords::CollectLayers visitor(cullRect, record, bounds, pictList, data); | |
| 819 for (int curOp = 0; curOp < record.count(); curOp++) { | |
| 820 visitor.setCurrentOp(curOp); | |
| 821 record.visit(curOp, visitor); | |
| 822 } | |
| 823 visitor.cleanUp(); | |
| 824 } | |
| OLD | NEW |