xref: /aosp_15_r20/external/skia/src/gpu/ganesh/GrStyle.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2016 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrStyle.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPath.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/utils/SkDashPathPriv.h"
12*c8dee2aaSAndroid Build Coastguard Worker 
13*c8dee2aaSAndroid Build Coastguard Worker #include <cstring>
14*c8dee2aaSAndroid Build Coastguard Worker 
KeySize(const GrStyle & style,Apply apply,uint32_t flags)15*c8dee2aaSAndroid Build Coastguard Worker int GrStyle::KeySize(const GrStyle &style, Apply apply, uint32_t flags) {
16*c8dee2aaSAndroid Build Coastguard Worker     static_assert(sizeof(uint32_t) == sizeof(SkScalar));
17*c8dee2aaSAndroid Build Coastguard Worker     int size = 0;
18*c8dee2aaSAndroid Build Coastguard Worker     if (style.isDashed()) {
19*c8dee2aaSAndroid Build Coastguard Worker         // One scalar for scale, one for dash phase, and one for each dash value.
20*c8dee2aaSAndroid Build Coastguard Worker         size += 2 + style.dashIntervalCnt();
21*c8dee2aaSAndroid Build Coastguard Worker     } else if (style.pathEffect()) {
22*c8dee2aaSAndroid Build Coastguard Worker         // No key for a generic path effect.
23*c8dee2aaSAndroid Build Coastguard Worker         return -1;
24*c8dee2aaSAndroid Build Coastguard Worker     }
25*c8dee2aaSAndroid Build Coastguard Worker 
26*c8dee2aaSAndroid Build Coastguard Worker     if (Apply::kPathEffectOnly == apply) {
27*c8dee2aaSAndroid Build Coastguard Worker         return size;
28*c8dee2aaSAndroid Build Coastguard Worker     }
29*c8dee2aaSAndroid Build Coastguard Worker 
30*c8dee2aaSAndroid Build Coastguard Worker     if (style.strokeRec().needToApply()) {
31*c8dee2aaSAndroid Build Coastguard Worker         // One for res scale, one for style/cap/join, one for miter limit, and one for width.
32*c8dee2aaSAndroid Build Coastguard Worker         size += 4;
33*c8dee2aaSAndroid Build Coastguard Worker     }
34*c8dee2aaSAndroid Build Coastguard Worker     return size;
35*c8dee2aaSAndroid Build Coastguard Worker }
36*c8dee2aaSAndroid Build Coastguard Worker 
WriteKey(uint32_t * key,const GrStyle & style,Apply apply,SkScalar scale,uint32_t flags)37*c8dee2aaSAndroid Build Coastguard Worker void GrStyle::WriteKey(uint32_t *key, const GrStyle &style, Apply apply, SkScalar scale,
38*c8dee2aaSAndroid Build Coastguard Worker                        uint32_t flags) {
39*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(key);
40*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(KeySize(style, apply) >= 0);
41*c8dee2aaSAndroid Build Coastguard Worker     static_assert(sizeof(uint32_t) == sizeof(SkScalar));
42*c8dee2aaSAndroid Build Coastguard Worker 
43*c8dee2aaSAndroid Build Coastguard Worker     int i = 0;
44*c8dee2aaSAndroid Build Coastguard Worker     // The scale can influence both the path effect and stroking. We want to preserve the
45*c8dee2aaSAndroid Build Coastguard Worker     // property that the following two are equal:
46*c8dee2aaSAndroid Build Coastguard Worker     // 1. WriteKey with apply == kPathEffectAndStrokeRec
47*c8dee2aaSAndroid Build Coastguard Worker     // 2. WriteKey with apply == kPathEffectOnly followed by WriteKey of a GrStyle made
48*c8dee2aaSAndroid Build Coastguard Worker     //    from SkStrokeRec output by the the path effect (and no additional path effect).
49*c8dee2aaSAndroid Build Coastguard Worker     // Since the scale can affect both parts of 2 we write it into the key twice.
50*c8dee2aaSAndroid Build Coastguard Worker     if (style.isDashed()) {
51*c8dee2aaSAndroid Build Coastguard Worker         static_assert(sizeof(style.dashPhase()) == sizeof(uint32_t));
52*c8dee2aaSAndroid Build Coastguard Worker         SkScalar phase = style.dashPhase();
53*c8dee2aaSAndroid Build Coastguard Worker         memcpy(&key[i++], &scale, sizeof(SkScalar));
54*c8dee2aaSAndroid Build Coastguard Worker         memcpy(&key[i++], &phase, sizeof(SkScalar));
55*c8dee2aaSAndroid Build Coastguard Worker 
56*c8dee2aaSAndroid Build Coastguard Worker         int32_t count = style.dashIntervalCnt();
57*c8dee2aaSAndroid Build Coastguard Worker         // Dash count should always be even.
58*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(0 == (count & 0x1));
59*c8dee2aaSAndroid Build Coastguard Worker         const SkScalar *intervals = style.dashIntervals();
60*c8dee2aaSAndroid Build Coastguard Worker         int intervalByteCnt = count * sizeof(SkScalar);
61*c8dee2aaSAndroid Build Coastguard Worker         memcpy(&key[i], intervals, intervalByteCnt);
62*c8dee2aaSAndroid Build Coastguard Worker         i += count;
63*c8dee2aaSAndroid Build Coastguard Worker     } else {
64*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!style.pathEffect());
65*c8dee2aaSAndroid Build Coastguard Worker     }
66*c8dee2aaSAndroid Build Coastguard Worker 
67*c8dee2aaSAndroid Build Coastguard Worker     if (Apply::kPathEffectAndStrokeRec == apply && style.strokeRec().needToApply()) {
68*c8dee2aaSAndroid Build Coastguard Worker         memcpy(&key[i++], &scale, sizeof(SkScalar));
69*c8dee2aaSAndroid Build Coastguard Worker         static constexpr uint32_t kStyleBits = 2;
70*c8dee2aaSAndroid Build Coastguard Worker         static constexpr uint32_t kJoinBits = 2;
71*c8dee2aaSAndroid Build Coastguard Worker         static constexpr uint32_t kCapBits = 32 - kStyleBits - kJoinBits;
72*c8dee2aaSAndroid Build Coastguard Worker         static constexpr uint32_t kJoinShift = kStyleBits;
73*c8dee2aaSAndroid Build Coastguard Worker         static constexpr uint32_t kCapShift = kJoinShift + kJoinBits;
74*c8dee2aaSAndroid Build Coastguard Worker 
75*c8dee2aaSAndroid Build Coastguard Worker         static_assert(SkStrokeRec::kStyleCount <= (1 << kStyleBits));
76*c8dee2aaSAndroid Build Coastguard Worker         static_assert(SkPaint::kJoinCount <= (1 << kJoinBits));
77*c8dee2aaSAndroid Build Coastguard Worker         static_assert(SkPaint::kCapCount <= (1 << kCapBits));
78*c8dee2aaSAndroid Build Coastguard Worker         // The cap type only matters for unclosed shapes. However, a path effect could unclose
79*c8dee2aaSAndroid Build Coastguard Worker         // the shape before it is stroked.
80*c8dee2aaSAndroid Build Coastguard Worker         SkPaint::Cap cap = SkPaint::kDefault_Cap;
81*c8dee2aaSAndroid Build Coastguard Worker         if (!(flags & kClosed_KeyFlag) || style.pathEffect()) {
82*c8dee2aaSAndroid Build Coastguard Worker             cap = style.strokeRec().getCap();
83*c8dee2aaSAndroid Build Coastguard Worker         }
84*c8dee2aaSAndroid Build Coastguard Worker         SkScalar miter = -1.f;
85*c8dee2aaSAndroid Build Coastguard Worker         SkPaint::Join join = SkPaint::kDefault_Join;
86*c8dee2aaSAndroid Build Coastguard Worker 
87*c8dee2aaSAndroid Build Coastguard Worker         // Dashing will not insert joins but other path effects may.
88*c8dee2aaSAndroid Build Coastguard Worker         if (!(flags & kNoJoins_KeyFlag) || style.hasNonDashPathEffect()) {
89*c8dee2aaSAndroid Build Coastguard Worker             join = style.strokeRec().getJoin();
90*c8dee2aaSAndroid Build Coastguard Worker             // Miter limit only affects miter joins
91*c8dee2aaSAndroid Build Coastguard Worker             if (SkPaint::kMiter_Join == join) {
92*c8dee2aaSAndroid Build Coastguard Worker                 miter = style.strokeRec().getMiter();
93*c8dee2aaSAndroid Build Coastguard Worker             }
94*c8dee2aaSAndroid Build Coastguard Worker         }
95*c8dee2aaSAndroid Build Coastguard Worker 
96*c8dee2aaSAndroid Build Coastguard Worker         key[i++] = style.strokeRec().getStyle() |
97*c8dee2aaSAndroid Build Coastguard Worker                    join << kJoinShift |
98*c8dee2aaSAndroid Build Coastguard Worker                    cap << kCapShift;
99*c8dee2aaSAndroid Build Coastguard Worker 
100*c8dee2aaSAndroid Build Coastguard Worker         memcpy(&key[i++], &miter, sizeof(miter));
101*c8dee2aaSAndroid Build Coastguard Worker 
102*c8dee2aaSAndroid Build Coastguard Worker         SkScalar width = style.strokeRec().getWidth();
103*c8dee2aaSAndroid Build Coastguard Worker         memcpy(&key[i++], &width, sizeof(width));
104*c8dee2aaSAndroid Build Coastguard Worker     }
105*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(KeySize(style, apply) == i);
106*c8dee2aaSAndroid Build Coastguard Worker }
107*c8dee2aaSAndroid Build Coastguard Worker 
initPathEffect(sk_sp<SkPathEffect> pe)108*c8dee2aaSAndroid Build Coastguard Worker void GrStyle::initPathEffect(sk_sp<SkPathEffect> pe) {
109*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(!fPathEffect);
110*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(SkPathEffectBase::DashType::kNone == fDashInfo.fType);
111*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(0 == fDashInfo.fIntervals.count());
112*c8dee2aaSAndroid Build Coastguard Worker     if (!pe) {
113*c8dee2aaSAndroid Build Coastguard Worker         return;
114*c8dee2aaSAndroid Build Coastguard Worker     }
115*c8dee2aaSAndroid Build Coastguard Worker     SkPathEffectBase::DashInfo info;
116*c8dee2aaSAndroid Build Coastguard Worker     if (SkPathEffectBase::DashType::kDash == as_PEB(pe)->asADash(&info)) {
117*c8dee2aaSAndroid Build Coastguard Worker         SkStrokeRec::Style recStyle = fStrokeRec.getStyle();
118*c8dee2aaSAndroid Build Coastguard Worker         if (recStyle != SkStrokeRec::kFill_Style && recStyle != SkStrokeRec::kStrokeAndFill_Style) {
119*c8dee2aaSAndroid Build Coastguard Worker             fDashInfo.fType = SkPathEffectBase::DashType::kDash;
120*c8dee2aaSAndroid Build Coastguard Worker             fDashInfo.fIntervals.reset(info.fCount);
121*c8dee2aaSAndroid Build Coastguard Worker             fDashInfo.fPhase = info.fPhase;
122*c8dee2aaSAndroid Build Coastguard Worker             info.fIntervals = fDashInfo.fIntervals.get();
123*c8dee2aaSAndroid Build Coastguard Worker             as_PEB(pe)->asADash(&info);
124*c8dee2aaSAndroid Build Coastguard Worker             fPathEffect = std::move(pe);
125*c8dee2aaSAndroid Build Coastguard Worker         }
126*c8dee2aaSAndroid Build Coastguard Worker     } else {
127*c8dee2aaSAndroid Build Coastguard Worker         fPathEffect = std::move(pe);
128*c8dee2aaSAndroid Build Coastguard Worker     }
129*c8dee2aaSAndroid Build Coastguard Worker }
130*c8dee2aaSAndroid Build Coastguard Worker 
applyPathEffect(SkPath * dst,SkStrokeRec * strokeRec,const SkPath & src) const131*c8dee2aaSAndroid Build Coastguard Worker bool GrStyle::applyPathEffect(SkPath* dst, SkStrokeRec* strokeRec, const SkPath& src) const {
132*c8dee2aaSAndroid Build Coastguard Worker     if (!fPathEffect) {
133*c8dee2aaSAndroid Build Coastguard Worker         return false;
134*c8dee2aaSAndroid Build Coastguard Worker     }
135*c8dee2aaSAndroid Build Coastguard Worker 
136*c8dee2aaSAndroid Build Coastguard Worker     // TODO: [skbug.com/11957] Plumb CTM callers and pass it to filterPath().
137*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(!fPathEffect->needsCTM());
138*c8dee2aaSAndroid Build Coastguard Worker 
139*c8dee2aaSAndroid Build Coastguard Worker     if (SkPathEffectBase::DashType::kDash == fDashInfo.fType) {
140*c8dee2aaSAndroid Build Coastguard Worker         // We apply the dash ourselves here rather than using the path effect. This is so that
141*c8dee2aaSAndroid Build Coastguard Worker         // we can control whether the dasher applies the strokeRec for special cases. Our keying
142*c8dee2aaSAndroid Build Coastguard Worker         // depends on the strokeRec being applied separately.
143*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!fPathEffect->needsCTM());  // Make sure specified PE doesn't need CTM
144*c8dee2aaSAndroid Build Coastguard Worker         SkScalar phase = fDashInfo.fPhase;
145*c8dee2aaSAndroid Build Coastguard Worker         const SkScalar* intervals = fDashInfo.fIntervals.get();
146*c8dee2aaSAndroid Build Coastguard Worker         int intervalCnt = fDashInfo.fIntervals.count();
147*c8dee2aaSAndroid Build Coastguard Worker         SkScalar initialLength;
148*c8dee2aaSAndroid Build Coastguard Worker         int initialIndex;
149*c8dee2aaSAndroid Build Coastguard Worker         SkScalar intervalLength;
150*c8dee2aaSAndroid Build Coastguard Worker         SkDashPath::CalcDashParameters(phase, intervals, intervalCnt, &initialLength,
151*c8dee2aaSAndroid Build Coastguard Worker                                        &initialIndex, &intervalLength);
152*c8dee2aaSAndroid Build Coastguard Worker         if (!SkDashPath::InternalFilter(dst, src, strokeRec,
153*c8dee2aaSAndroid Build Coastguard Worker                                         nullptr, intervals, intervalCnt,
154*c8dee2aaSAndroid Build Coastguard Worker                                         initialLength, initialIndex, intervalLength, phase,
155*c8dee2aaSAndroid Build Coastguard Worker                                         SkDashPath::StrokeRecApplication::kDisallow)) {
156*c8dee2aaSAndroid Build Coastguard Worker             return false;
157*c8dee2aaSAndroid Build Coastguard Worker         }
158*c8dee2aaSAndroid Build Coastguard Worker     } else if (!fPathEffect->filterPath(dst, src, strokeRec, nullptr)) {
159*c8dee2aaSAndroid Build Coastguard Worker         return false;
160*c8dee2aaSAndroid Build Coastguard Worker     }
161*c8dee2aaSAndroid Build Coastguard Worker     dst->setIsVolatile(true);
162*c8dee2aaSAndroid Build Coastguard Worker     return true;
163*c8dee2aaSAndroid Build Coastguard Worker }
164*c8dee2aaSAndroid Build Coastguard Worker 
applyPathEffectToPath(SkPath * dst,SkStrokeRec * remainingStroke,const SkPath & src,SkScalar resScale) const165*c8dee2aaSAndroid Build Coastguard Worker bool GrStyle::applyPathEffectToPath(SkPath *dst, SkStrokeRec *remainingStroke,
166*c8dee2aaSAndroid Build Coastguard Worker                                     const SkPath &src, SkScalar resScale) const {
167*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dst);
168*c8dee2aaSAndroid Build Coastguard Worker     SkStrokeRec strokeRec = fStrokeRec;
169*c8dee2aaSAndroid Build Coastguard Worker     strokeRec.setResScale(resScale);
170*c8dee2aaSAndroid Build Coastguard Worker     if (!this->applyPathEffect(dst, &strokeRec, src)) {
171*c8dee2aaSAndroid Build Coastguard Worker         return false;
172*c8dee2aaSAndroid Build Coastguard Worker     }
173*c8dee2aaSAndroid Build Coastguard Worker     *remainingStroke = strokeRec;
174*c8dee2aaSAndroid Build Coastguard Worker     return true;
175*c8dee2aaSAndroid Build Coastguard Worker }
176*c8dee2aaSAndroid Build Coastguard Worker 
applyToPath(SkPath * dst,SkStrokeRec::InitStyle * style,const SkPath & src,SkScalar resScale) const177*c8dee2aaSAndroid Build Coastguard Worker bool GrStyle::applyToPath(SkPath* dst, SkStrokeRec::InitStyle* style, const SkPath& src,
178*c8dee2aaSAndroid Build Coastguard Worker                           SkScalar resScale) const {
179*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(style);
180*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dst);
181*c8dee2aaSAndroid Build Coastguard Worker     SkStrokeRec strokeRec = fStrokeRec;
182*c8dee2aaSAndroid Build Coastguard Worker     strokeRec.setResScale(resScale);
183*c8dee2aaSAndroid Build Coastguard Worker     const SkPath* pathForStrokeRec = &src;
184*c8dee2aaSAndroid Build Coastguard Worker     if (this->applyPathEffect(dst, &strokeRec, src)) {
185*c8dee2aaSAndroid Build Coastguard Worker         pathForStrokeRec = dst;
186*c8dee2aaSAndroid Build Coastguard Worker     } else if (fPathEffect) {
187*c8dee2aaSAndroid Build Coastguard Worker         return false;
188*c8dee2aaSAndroid Build Coastguard Worker     }
189*c8dee2aaSAndroid Build Coastguard Worker     if (strokeRec.needToApply()) {
190*c8dee2aaSAndroid Build Coastguard Worker         if (!strokeRec.applyToPath(dst, *pathForStrokeRec)) {
191*c8dee2aaSAndroid Build Coastguard Worker             return false;
192*c8dee2aaSAndroid Build Coastguard Worker         }
193*c8dee2aaSAndroid Build Coastguard Worker         dst->setIsVolatile(true);
194*c8dee2aaSAndroid Build Coastguard Worker         *style = SkStrokeRec::kFill_InitStyle;
195*c8dee2aaSAndroid Build Coastguard Worker     } else if (!fPathEffect) {
196*c8dee2aaSAndroid Build Coastguard Worker         // Nothing to do for path effect or stroke, fail.
197*c8dee2aaSAndroid Build Coastguard Worker         return false;
198*c8dee2aaSAndroid Build Coastguard Worker     } else {
199*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(SkStrokeRec::kFill_Style == strokeRec.getStyle() ||
200*c8dee2aaSAndroid Build Coastguard Worker                  SkStrokeRec::kHairline_Style == strokeRec.getStyle());
201*c8dee2aaSAndroid Build Coastguard Worker         *style = strokeRec.getStyle() == SkStrokeRec::kFill_Style
202*c8dee2aaSAndroid Build Coastguard Worker                  ? SkStrokeRec::kFill_InitStyle
203*c8dee2aaSAndroid Build Coastguard Worker                  : SkStrokeRec::kHairline_InitStyle;
204*c8dee2aaSAndroid Build Coastguard Worker     }
205*c8dee2aaSAndroid Build Coastguard Worker     return true;
206*c8dee2aaSAndroid Build Coastguard Worker }
207