xref: /aosp_15_r20/external/skia/fuzz/FuzzPathop.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 "fuzz/Fuzz.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "fuzz/FuzzCommon.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPath.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/pathops/SkPathOps.h"
13*c8dee2aaSAndroid Build Coastguard Worker 
14*c8dee2aaSAndroid Build Coastguard Worker const uint8_t MAX_OPS = 20;
15*c8dee2aaSAndroid Build Coastguard Worker 
DEF_FUZZ(Pathop,fuzz)16*c8dee2aaSAndroid Build Coastguard Worker DEF_FUZZ(Pathop, fuzz) {
17*c8dee2aaSAndroid Build Coastguard Worker 
18*c8dee2aaSAndroid Build Coastguard Worker     uint8_t choice;
19*c8dee2aaSAndroid Build Coastguard Worker     fuzz->nextRange(&choice, 0, 4);
20*c8dee2aaSAndroid Build Coastguard Worker     switch (choice) {
21*c8dee2aaSAndroid Build Coastguard Worker         case 0: {
22*c8dee2aaSAndroid Build Coastguard Worker             uint8_t ops;
23*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&ops, 0, MAX_OPS);
24*c8dee2aaSAndroid Build Coastguard Worker             SkOpBuilder builder;
25*c8dee2aaSAndroid Build Coastguard Worker             for (uint8_t i = 0; i < ops && !fuzz->exhausted(); i++) {
26*c8dee2aaSAndroid Build Coastguard Worker                 SkPath path;
27*c8dee2aaSAndroid Build Coastguard Worker                 FuzzEvilPath(fuzz, &path, SkPath::Verb::kDone_Verb);
28*c8dee2aaSAndroid Build Coastguard Worker                 SkPathFillType ft;
29*c8dee2aaSAndroid Build Coastguard Worker                 fuzz->nextRange(&ft, 0, (int)SkPathFillType::kInverseEvenOdd);
30*c8dee2aaSAndroid Build Coastguard Worker                 path.setFillType(ft);
31*c8dee2aaSAndroid Build Coastguard Worker 
32*c8dee2aaSAndroid Build Coastguard Worker                 SkPathOp op;
33*c8dee2aaSAndroid Build Coastguard Worker                 fuzz->nextRange(&op, 0, SkPathOp::kReverseDifference_SkPathOp);
34*c8dee2aaSAndroid Build Coastguard Worker                 builder.add(path, op);
35*c8dee2aaSAndroid Build Coastguard Worker             }
36*c8dee2aaSAndroid Build Coastguard Worker 
37*c8dee2aaSAndroid Build Coastguard Worker             SkPath result;
38*c8dee2aaSAndroid Build Coastguard Worker             builder.resolve(&result);
39*c8dee2aaSAndroid Build Coastguard Worker             break;
40*c8dee2aaSAndroid Build Coastguard Worker         }
41*c8dee2aaSAndroid Build Coastguard Worker         case 1: {
42*c8dee2aaSAndroid Build Coastguard Worker             SkPath path;
43*c8dee2aaSAndroid Build Coastguard Worker             FuzzEvilPath(fuzz, &path, SkPath::Verb::kDone_Verb);
44*c8dee2aaSAndroid Build Coastguard Worker             SkPathFillType ft;
45*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&ft, 0, (int)SkPathFillType::kInverseEvenOdd);
46*c8dee2aaSAndroid Build Coastguard Worker             path.setFillType(ft);
47*c8dee2aaSAndroid Build Coastguard Worker 
48*c8dee2aaSAndroid Build Coastguard Worker             SkPath result;
49*c8dee2aaSAndroid Build Coastguard Worker             bool isSame;
50*c8dee2aaSAndroid Build Coastguard Worker             fuzz->next(&isSame);
51*c8dee2aaSAndroid Build Coastguard Worker             if (isSame) {
52*c8dee2aaSAndroid Build Coastguard Worker                 result = path;
53*c8dee2aaSAndroid Build Coastguard Worker             }
54*c8dee2aaSAndroid Build Coastguard Worker             Simplify(path, &result);
55*c8dee2aaSAndroid Build Coastguard Worker             break;
56*c8dee2aaSAndroid Build Coastguard Worker         }
57*c8dee2aaSAndroid Build Coastguard Worker         case 2: {
58*c8dee2aaSAndroid Build Coastguard Worker             SkPath path;
59*c8dee2aaSAndroid Build Coastguard Worker             FuzzEvilPath(fuzz, &path, SkPath::Verb::kDone_Verb);
60*c8dee2aaSAndroid Build Coastguard Worker             SkPathFillType ft;
61*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&ft, 0, SkPathFillType::kInverseEvenOdd);
62*c8dee2aaSAndroid Build Coastguard Worker             path.setFillType(ft);
63*c8dee2aaSAndroid Build Coastguard Worker 
64*c8dee2aaSAndroid Build Coastguard Worker             SkPath path2;
65*c8dee2aaSAndroid Build Coastguard Worker             FuzzEvilPath(fuzz, &path2, SkPath::Verb::kDone_Verb);
66*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&ft, 0, SkPathFillType::kInverseEvenOdd);
67*c8dee2aaSAndroid Build Coastguard Worker             path.setFillType(ft);
68*c8dee2aaSAndroid Build Coastguard Worker 
69*c8dee2aaSAndroid Build Coastguard Worker             SkPathOp op;
70*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&op, 0, SkPathOp::kReverseDifference_SkPathOp);
71*c8dee2aaSAndroid Build Coastguard Worker 
72*c8dee2aaSAndroid Build Coastguard Worker             SkPath result;
73*c8dee2aaSAndroid Build Coastguard Worker             uint8_t pickOutput;
74*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&pickOutput, 0, 2);
75*c8dee2aaSAndroid Build Coastguard Worker             if (pickOutput == 1) {
76*c8dee2aaSAndroid Build Coastguard Worker                 result = path;
77*c8dee2aaSAndroid Build Coastguard Worker             } else if (pickOutput == 2) {
78*c8dee2aaSAndroid Build Coastguard Worker                 result = path2;
79*c8dee2aaSAndroid Build Coastguard Worker             }
80*c8dee2aaSAndroid Build Coastguard Worker             Op(path, path2, op, &result);
81*c8dee2aaSAndroid Build Coastguard Worker             break;
82*c8dee2aaSAndroid Build Coastguard Worker         }
83*c8dee2aaSAndroid Build Coastguard Worker         case 3: {
84*c8dee2aaSAndroid Build Coastguard Worker             SkPath path;
85*c8dee2aaSAndroid Build Coastguard Worker             FuzzEvilPath(fuzz, &path, SkPath::Verb::kDone_Verb);
86*c8dee2aaSAndroid Build Coastguard Worker             SkPathFillType ft;
87*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&ft, 0, SkPathFillType::kInverseEvenOdd);
88*c8dee2aaSAndroid Build Coastguard Worker             path.setFillType(ft);
89*c8dee2aaSAndroid Build Coastguard Worker 
90*c8dee2aaSAndroid Build Coastguard Worker             SkPath result;
91*c8dee2aaSAndroid Build Coastguard Worker             bool isSame;
92*c8dee2aaSAndroid Build Coastguard Worker             fuzz->next(&isSame);
93*c8dee2aaSAndroid Build Coastguard Worker             if (isSame) {
94*c8dee2aaSAndroid Build Coastguard Worker                 result = path;
95*c8dee2aaSAndroid Build Coastguard Worker             }
96*c8dee2aaSAndroid Build Coastguard Worker             AsWinding(path, &result);
97*c8dee2aaSAndroid Build Coastguard Worker             break;
98*c8dee2aaSAndroid Build Coastguard Worker         }
99*c8dee2aaSAndroid Build Coastguard Worker         case 4: {
100*c8dee2aaSAndroid Build Coastguard Worker             SkPath path;
101*c8dee2aaSAndroid Build Coastguard Worker             FuzzEvilPath(fuzz, &path, SkPath::Verb::kDone_Verb);
102*c8dee2aaSAndroid Build Coastguard Worker             SkPathFillType ft;
103*c8dee2aaSAndroid Build Coastguard Worker             fuzz->nextRange(&ft, 0, SkPathFillType::kInverseEvenOdd);
104*c8dee2aaSAndroid Build Coastguard Worker             path.setFillType(ft);
105*c8dee2aaSAndroid Build Coastguard Worker 
106*c8dee2aaSAndroid Build Coastguard Worker             SkRect result;
107*c8dee2aaSAndroid Build Coastguard Worker             TightBounds(path, &result);
108*c8dee2aaSAndroid Build Coastguard Worker             break;
109*c8dee2aaSAndroid Build Coastguard Worker         }
110*c8dee2aaSAndroid Build Coastguard Worker         default: {
111*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(false);
112*c8dee2aaSAndroid Build Coastguard Worker             break;
113*c8dee2aaSAndroid Build Coastguard Worker         }
114*c8dee2aaSAndroid Build Coastguard Worker     }
115*c8dee2aaSAndroid Build Coastguard Worker }
116*c8dee2aaSAndroid Build Coastguard Worker 
117*c8dee2aaSAndroid Build Coastguard Worker 
118*c8dee2aaSAndroid Build Coastguard Worker const int kLastOp = SkPathOp::kReverseDifference_SkPathOp;
119*c8dee2aaSAndroid Build Coastguard Worker 
BuildPath(Fuzz * fuzz,SkPath * path)120*c8dee2aaSAndroid Build Coastguard Worker void BuildPath(Fuzz* fuzz, SkPath* path) {
121*c8dee2aaSAndroid Build Coastguard Worker     while (!fuzz->exhausted()) {
122*c8dee2aaSAndroid Build Coastguard Worker     // Use a uint8_t to conserve bytes.  This makes our "fuzzed bytes footprint"
123*c8dee2aaSAndroid Build Coastguard Worker     // smaller, which leads to more efficient fuzzing.
124*c8dee2aaSAndroid Build Coastguard Worker     uint8_t operation;
125*c8dee2aaSAndroid Build Coastguard Worker     fuzz->next(&operation);
126*c8dee2aaSAndroid Build Coastguard Worker     SkScalar a,b,c,d,e,f;
127*c8dee2aaSAndroid Build Coastguard Worker 
128*c8dee2aaSAndroid Build Coastguard Worker     switch (operation % (SkPath::Verb::kDone_Verb + 1)) {
129*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kMove_Verb:
130*c8dee2aaSAndroid Build Coastguard Worker         if (fuzz->remainingSize() < (2*sizeof(SkScalar))) {
131*c8dee2aaSAndroid Build Coastguard Worker             fuzz->deplete();
132*c8dee2aaSAndroid Build Coastguard Worker             return;
133*c8dee2aaSAndroid Build Coastguard Worker         }
134*c8dee2aaSAndroid Build Coastguard Worker         fuzz->next(&a, &b);
135*c8dee2aaSAndroid Build Coastguard Worker         path->moveTo(a, b);
136*c8dee2aaSAndroid Build Coastguard Worker         break;
137*c8dee2aaSAndroid Build Coastguard Worker 
138*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kLine_Verb:
139*c8dee2aaSAndroid Build Coastguard Worker         if (fuzz->remainingSize() < (2*sizeof(SkScalar))) {
140*c8dee2aaSAndroid Build Coastguard Worker             fuzz->deplete();
141*c8dee2aaSAndroid Build Coastguard Worker             return;
142*c8dee2aaSAndroid Build Coastguard Worker         }
143*c8dee2aaSAndroid Build Coastguard Worker         fuzz->next(&a, &b);
144*c8dee2aaSAndroid Build Coastguard Worker         path->lineTo(a, b);
145*c8dee2aaSAndroid Build Coastguard Worker         break;
146*c8dee2aaSAndroid Build Coastguard Worker 
147*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kQuad_Verb:
148*c8dee2aaSAndroid Build Coastguard Worker         if (fuzz->remainingSize() < (4*sizeof(SkScalar))) {
149*c8dee2aaSAndroid Build Coastguard Worker             fuzz->deplete();
150*c8dee2aaSAndroid Build Coastguard Worker             return;
151*c8dee2aaSAndroid Build Coastguard Worker         }
152*c8dee2aaSAndroid Build Coastguard Worker         fuzz->next(&a, &b, &c, &d);
153*c8dee2aaSAndroid Build Coastguard Worker         path->quadTo(a, b, c, d);
154*c8dee2aaSAndroid Build Coastguard Worker         break;
155*c8dee2aaSAndroid Build Coastguard Worker 
156*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kConic_Verb:
157*c8dee2aaSAndroid Build Coastguard Worker         if (fuzz->remainingSize() < (5*sizeof(SkScalar))) {
158*c8dee2aaSAndroid Build Coastguard Worker             fuzz->deplete();
159*c8dee2aaSAndroid Build Coastguard Worker             return;
160*c8dee2aaSAndroid Build Coastguard Worker         }
161*c8dee2aaSAndroid Build Coastguard Worker         fuzz->next(&a, &b, &c, &d, &e);
162*c8dee2aaSAndroid Build Coastguard Worker         path->conicTo(a, b, c, d, e);
163*c8dee2aaSAndroid Build Coastguard Worker         break;
164*c8dee2aaSAndroid Build Coastguard Worker 
165*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kCubic_Verb:
166*c8dee2aaSAndroid Build Coastguard Worker         if (fuzz->remainingSize() < (6*sizeof(SkScalar))) {
167*c8dee2aaSAndroid Build Coastguard Worker             fuzz->deplete();
168*c8dee2aaSAndroid Build Coastguard Worker             return;
169*c8dee2aaSAndroid Build Coastguard Worker         }
170*c8dee2aaSAndroid Build Coastguard Worker         fuzz->next(&a, &b, &c, &d, &e, &f);
171*c8dee2aaSAndroid Build Coastguard Worker         path->cubicTo(a, b, c, d, e, f);
172*c8dee2aaSAndroid Build Coastguard Worker         break;
173*c8dee2aaSAndroid Build Coastguard Worker 
174*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kClose_Verb:
175*c8dee2aaSAndroid Build Coastguard Worker         path->close();
176*c8dee2aaSAndroid Build Coastguard Worker         break;
177*c8dee2aaSAndroid Build Coastguard Worker 
178*c8dee2aaSAndroid Build Coastguard Worker       case SkPath::Verb::kDone_Verb:
179*c8dee2aaSAndroid Build Coastguard Worker         // In this case, simply exit.
180*c8dee2aaSAndroid Build Coastguard Worker         return;
181*c8dee2aaSAndroid Build Coastguard Worker     }
182*c8dee2aaSAndroid Build Coastguard Worker   }
183*c8dee2aaSAndroid Build Coastguard Worker }
184*c8dee2aaSAndroid Build Coastguard Worker 
DEF_FUZZ(LegacyChromiumPathop,fuzz)185*c8dee2aaSAndroid Build Coastguard Worker DEF_FUZZ(LegacyChromiumPathop, fuzz) {
186*c8dee2aaSAndroid Build Coastguard Worker     // See https://cs.chromium.org/chromium/src/testing/libfuzzer/fuzzers/skia_pathop_fuzzer.cc
187*c8dee2aaSAndroid Build Coastguard Worker     SkOpBuilder builder;
188*c8dee2aaSAndroid Build Coastguard Worker     while (!fuzz->exhausted()) {
189*c8dee2aaSAndroid Build Coastguard Worker         SkPath path;
190*c8dee2aaSAndroid Build Coastguard Worker         uint8_t op;
191*c8dee2aaSAndroid Build Coastguard Worker         fuzz->next(&op);
192*c8dee2aaSAndroid Build Coastguard Worker         if (fuzz->exhausted()) {
193*c8dee2aaSAndroid Build Coastguard Worker             break;
194*c8dee2aaSAndroid Build Coastguard Worker         }
195*c8dee2aaSAndroid Build Coastguard Worker 
196*c8dee2aaSAndroid Build Coastguard Worker         BuildPath(fuzz, &path);
197*c8dee2aaSAndroid Build Coastguard Worker         builder.add(path, static_cast<SkPathOp>(op % (kLastOp + 1)));
198*c8dee2aaSAndroid Build Coastguard Worker     }
199*c8dee2aaSAndroid Build Coastguard Worker 
200*c8dee2aaSAndroid Build Coastguard Worker     SkPath result;
201*c8dee2aaSAndroid Build Coastguard Worker     builder.resolve(&result);
202*c8dee2aaSAndroid Build Coastguard Worker }
203