Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(179)

Side by Side Diff: core/cross/gpu2d/path_processor.cc

Issue 652016: Added the bulk of the algorithm for GPU accelerated 2D vector curve... (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/o3d/
Patch Set: '' Created 10 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « core/cross/gpu2d/path_processor.h ('k') | core/cross/primitive.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Property Changes:
Added: svn:eol-style
+ LF
OLDNEW
(Empty)
1 /*
2 * Copyright 2010, Google Inc.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are
7 * met:
8 *
9 * * Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above
12 * copyright notice, this list of conditions and the following disclaimer
13 * in the documentation and/or other materials provided with the
14 * distribution.
15 * * Neither the name of Google Inc. nor the names of its
16 * contributors may be used to endorse or promote products derived from
17 * this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 */
31
32 #include "core/cross/gpu2d/path_processor.h"
33
34 #include <algorithm>
35
36 #include "base/logging.h"
37 #include "core/cross/gpu2d/arena.h"
38 #include "core/cross/gpu2d/cubic_classifier.h"
39 #include "core/cross/gpu2d/cubic_math_utils.h"
40 #include "core/cross/gpu2d/cubic_texture_coords.h"
41 #include "core/cross/gpu2d/interval_tree.h"
42 #include "core/cross/gpu2d/local_triangulator.h"
43 #include "core/cross/gpu2d/path_cache.h"
44 #include "third_party/skia/include/core/SkGeometry.h"
45 #include "third_party/skia/include/core/SkPath.h"
46 #include "third_party/skia/include/core/SkScalar.h"
47 #include "third_party/glu/internal_glu.h"
48
49 namespace o3d {
50 namespace gpu2d {
51
52 class Contour;
53
54 //----------------------------------------------------------------------
55 // min / max helpers
56 //
57
58 template <typename T>
59 T min2(const T& v1, const T& v2) {
60 return std::min(v1, v2);
61 }
62
63 template <typename T>
64 T max2(const T& v1, const T& v2) {
65 return std::max(v1, v2);
66 }
67
68 template <typename T>
69 T min3(const T& v1, const T& v2, const T& v3) {
70 return min2(min2(v1, v2), v3);
71 }
72
73 template <typename T>
74 T max3(const T& v1, const T& v2, const T& v3) {
75 return max2(max2(v1, v2), v3);
76 }
77
78 template <typename T>
79 T min4(const T& v1, const T& v2, const T& v3, const T& v4) {
80 return min2(min2(v1, v2), min2(v3, v4));
81 }
82
83 template <typename T>
84 T max4(const T& v1, const T& v2, const T& v3, const T& v4) {
85 return max2(max2(v1, v2), max2(v3, v4));
86 }
87
88 //----------------------------------------------------------------------
89 // BBox
90 //
91
92 // Extremely simple bounding box class for Segments.
93 class BBox {
94 public:
95 BBox()
96 : min_x_(0),
97 min_y_(0),
98 max_x_(0),
99 max_y_(0) {
100 }
101
102 // Initializes the parameters of the bounding box.
103 void Setup(float min_x,
104 float min_y,
105 float max_x,
106 float max_y) {
107 min_x_ = min_x;
108 min_y_ = min_y;
109 max_x_ = max_x;
110 max_y_ = max_y;
111 }
112
113 // Initializes the bounding box to surround the given triangle.
114 void Setup(LocalTriangulator::Triangle* triangle) {
115 Setup(min3(triangle->get_vertex(0)->x(),
116 triangle->get_vertex(1)->x(),
117 triangle->get_vertex(2)->x()),
118 min3(triangle->get_vertex(0)->y(),
119 triangle->get_vertex(1)->y(),
120 triangle->get_vertex(2)->y()),
121 max3(triangle->get_vertex(0)->x(),
122 triangle->get_vertex(1)->x(),
123 triangle->get_vertex(2)->x()),
124 max3(triangle->get_vertex(0)->y(),
125 triangle->get_vertex(1)->y(),
126 triangle->get_vertex(2)->y()));
127 }
128
129 float min_x() const { return min_x_; }
130 float min_y() const { return min_y_; }
131 float max_x() const { return max_x_; }
132 float max_y() const { return max_y_; }
133
134 private:
135 float min_x_;
136 float min_y_;
137 float max_x_;
138 float max_y_;
139 DISALLOW_COPY_AND_ASSIGN(BBox);
140 };
141
142 //----------------------------------------------------------------------
143 // Segment
144 //
145
146 // Describes a segment of the path: either a cubic or a line segment.
147 // These are stored in a doubly linked list to speed up curve
148 // subdivision, which occurs due to either rendering artifacts in the
149 // loop case or due to overlapping triangles.
150 class Segment {
151 public:
152 // The kind of segment: cubic or line.
153 enum Kind {
154 kCubic,
155 kLine
156 };
157
158 // No-argument constructor allows construction by the Arena class.
159 Segment()
160 : arena_(NULL),
161 kind_(kCubic),
162 prev_(NULL),
163 next_(NULL),
164 contour_(NULL),
165 triangulator_(NULL),
166 marked_for_subdivision_(false) {
167 }
168
169 // Initializer for cubic curve segments.
170 void Setup(Arena* arena,
171 Contour* contour,
172 SkPoint cp0,
173 SkPoint cp1,
174 SkPoint cp2,
175 SkPoint cp3) {
176 arena_ = arena;
177 contour_ = contour;
178 kind_ = kCubic;
179 points_[0] = cp0;
180 points_[1] = cp1;
181 points_[2] = cp2;
182 points_[3] = cp3;
183 ComputeBoundingBox();
184 }
185
186 // Initializer for line segments.
187 void Setup(Arena* arena,
188 Contour* contour,
189 SkPoint p0,
190 SkPoint p1) {
191 arena_ = arena;
192 contour_ = contour;
193 kind_ = kLine;
194 points_[0] = p0;
195 points_[1] = p1;
196 ComputeBoundingBox();
197 }
198
199 // Returns the kind of segment.
200 Kind kind() const {
201 return kind_;
202 }
203
204 // Returns the i'th control point, 0 <= i < 4.
205 const SkPoint& get_point(int i) {
206 DCHECK(i >= 0 && i < 4);
207 return points_[i];
208 }
209
210 // Returns the next segment in the contour.
211 Segment* next() const { return next_; }
212
213 // Returns the previous segment in the contour.
214 Segment* prev() const { return prev_; }
215
216 // Sets the next segment in the contour.
217 void set_next(Segment* next) { next_ = next; }
218
219 // Sets the previous segment in the contour.
220 void set_prev(Segment* prev) { prev_ = prev; }
221
222 // The contour this segment belongs to.
223 Contour* contour() const { return contour_; }
224
225 // Subdivides the current segment at the given parameter value (0 <=
226 // t <= 1) and replaces it with the two newly created Segments in
227 // the linked list, if possible. Returns a pointer to the leftmost
228 // Segment.
229 Segment* Subdivide(float param) {
230 SkPoint dst[7];
231 SkChopCubicAt(points_, dst, param);
232 Segment* left = arena_->Alloc<Segment>();
233 Segment* right = arena_->Alloc<Segment>();
234 left->Setup(arena_, contour_, dst[0], dst[1], dst[2], dst[3]);
235 right->Setup(arena_, contour_, dst[3], dst[4], dst[5], dst[6]);
236 left->set_next(right);
237 right->set_prev(left);
238 // Try to set up a link between "this->prev()" and "left".
239 if (prev() != NULL) {
240 left->set_prev(prev());
241 prev()->set_next(left);
242 }
243 // Try to set up a link between "this->next()" and "right".
244 Segment* n = next();
245 if (n != NULL) {
246 right->set_next(n);
247 n->set_prev(right);
248 }
249 // Set up a link between "this" and "left"; this is only to
250 // provide a certain amount of continuity during forward iteration.
251 set_next(left);
252 return left;
253 }
254
255 // Subdivides the current segment at the halfway point and replaces
256 // it with the two newly created Segments in the linked list, if
257 // possible. Returns a pointer to the leftmost Segment.
258 Segment* Subdivide() {
259 return Subdivide(0.5f);
260 }
261
262 // Returns the bounding box of this segment.
263 const BBox& bbox() const {
264 return bbox_;
265 }
266
267 // Computes the number of times a query line starting at the given
268 // point and extending to x=+infinity crosses this segment.
269 int NumCrossingsForXRay(SkPoint pt) const {
270 if (kind_ == kCubic) {
271 // Should consider caching the monotonic cubics.
272 return SkNumXRayCrossingsForCubic(pt, points_);
273 } else {
274 return SkXRayCrossesLine(pt, points_) ? 1 : 0;
275 }
276 }
277
278 // Performs a local triangulation of the control points in this
279 // segment. This operation only makes sense for cubic type segments.
280 void Triangulate(bool compute_inside_edges,
281 CubicTextureCoords::Result* tex_coords);
282
283 // Returns the number of control point triangles associated with
284 // this segment.
285 int num_triangles() const {
286 if (!triangulator_)
287 return 0;
288 return triangulator_->num_triangles();
289 }
290
291 // Fetches the given control point triangle for this segment.
292 LocalTriangulator::Triangle* get_triangle(int index) {
293 DCHECK(triangulator_);
294 return triangulator_->get_triangle(index);
295 }
296
297 // Number of vertices along the inside edge of this segment. This
298 // can be called either for line or cubic type segments.
299 int num_interior_vertices() const {
300 if (kind_ == kCubic) {
301 if (triangulator_) {
302 return triangulator_->num_interior_vertices();
303 } else {
304 return 0;
305 }
306 }
307
308 return 2;
309 }
310
311 // Returns the given interior vertex, 0 <= index < num_interior_vertices().
312 SkPoint get_interior_vertex(int index) const {
313 DCHECK(index >= 0 && index < num_interior_vertices());
314 if (kind_ == kCubic) {
315 SkPoint res;
316 if (triangulator_) {
317 LocalTriangulator::Vertex* vertex =
318 triangulator_->get_interior_vertex(index);
319 if (vertex)
320 res.set(SkFloatToScalar(vertex->x()),
321 SkFloatToScalar(vertex->y()));
322 }
323 return res;
324 }
325
326 return points_[index];
327 }
328
329 // State to assist with curve subdivision.
330 bool marked_for_subdivision() {
331 return marked_for_subdivision_;
332 }
333
334 // State to assist with curve subdivision.
335 void set_marked_for_subdivision(bool marked_for_subdivision) {
336 marked_for_subdivision_ = marked_for_subdivision;
337 }
338
339 private:
340 // Computes the bounding box of this Segment.
341 void ComputeBoundingBox() {
342 switch (kind_) {
343 case kCubic:
344 bbox_.Setup(SkScalarToFloat(min4(points_[0].fX,
345 points_[1].fX,
346 points_[2].fX,
347 points_[3].fX)),
348 SkScalarToFloat(min4(points_[0].fY,
349 points_[1].fY,
350 points_[2].fY,
351 points_[3].fY)),
352 SkScalarToFloat(max4(points_[0].fX,
353 points_[1].fX,
354 points_[2].fX,
355 points_[3].fX)),
356 SkScalarToFloat(max4(points_[0].fY,
357 points_[1].fY,
358 points_[2].fY,
359 points_[3].fY)));
360 break;
361
362 case kLine:
363 bbox_.Setup(SkScalarToFloat(min2(points_[0].fX,
364 points_[1].fX)),
365 SkScalarToFloat(min2(points_[0].fY,
366 points_[1].fY)),
367 SkScalarToFloat(max2(points_[0].fX,
368 points_[1].fX)),
369 SkScalarToFloat(max2(points_[0].fY,
370 points_[1].fY)));
371 break;
372
373 default:
374 NOTREACHED();
375 break;
376 }
377 }
378
379 Arena* arena_;
380 Kind kind_;
381 SkPoint points_[4];
382 Segment* prev_;
383 Segment* next_;
384 Contour* contour_;
385 BBox bbox_;
386 LocalTriangulator* triangulator_;
387 bool marked_for_subdivision_;
388
389 DISALLOW_COPY_AND_ASSIGN(Segment);
390 };
391
392 //----------------------------------------------------------------------
393 // Contour
394 //
395
396 // Describes a closed contour of the path.
397 class Contour {
398 public:
399 Contour() {
400 first_ = &sentinel_;
401 first_->set_next(first_);
402 first_->set_prev(first_);
403 ccw_ = true;
404 fill_right_side_ = true;
405 }
406
407 // Adds a segment to this contour.
408 void Add(Segment* segment) {
409 if (first_ == &sentinel_) {
410 // First element is the sentinel. Replace it with the incoming
411 // segment.
412 segment->set_next(first_);
413 segment->set_prev(first_);
414 first_->set_next(segment);
415 first_->set_prev(segment);
416 first_ = segment;
417 } else {
418 // first_->prev() is the sentinel.
419 Segment* sentinel = first_->prev();
420 Segment* last = sentinel->prev();
421 last->set_next(segment);
422 segment->set_prev(last);
423 segment->set_next(sentinel);
424 sentinel->set_prev(segment);
425 }
426 }
427
428 // Subdivides the given segment at the given parametric value.
429 // Returns a pointer to the first of the two portions of the
430 // subdivided segment.
431 Segment* Subdivide(Segment* segment, float param) {
432 Segment* left = segment->Subdivide(param);
433 if (first_ == segment)
434 first_ = left;
435 return left;
436 }
437
438 // Subdivides the given segment at the halfway point. Returns a
439 // pointer to the first of the two portions of the subdivided
440 // segment.
441 Segment* Subdivide(Segment* segment) {
442 Segment* left = segment->Subdivide();
443 if (first_ == segment)
444 first_ = left;
445 return left;
446 }
447
448 // Returns the first segment in the contour for iteration.
449 Segment* begin() const {
450 return first_;
451 }
452
453 // Returns the last segment in the contour for iteration. Callers
454 // should not iterate over this segment. In other words:
455 // for (Segment* cur = contour->begin();
456 // cur != contour->end();
457 // cur = cur->next()) {
458 // // .. process cur ...
459 // }
460 Segment* end() const {
461 return first_->prev();
462 }
463
464 // Returns whether this contour is oriented counterclockwise.
465 bool ccw() const {
466 return ccw_;
467 }
468
469 void set_ccw(bool ccw) {
470 ccw_ = ccw;
471 }
472
473 // Returns whether the right side of this contour is filled.
474 bool fill_right_side() const { return fill_right_side_; }
475
476 void set_fill_right_side(bool fill_right_side) {
477 fill_right_side_ = fill_right_side;
478 }
479
480 private:
481 // The start of the segment chain. The segments are kept in a
482 // circular doubly linked list for rapid access to the beginning and
483 // end.
484 Segment* first_;
485
486 // The sentinel element at the end of the chain, needed for
487 // reasonable iteration semantics.
488 Segment sentinel_;
489
490 // Whether this contour is oriented counterclockwise.
491 bool ccw_;
492
493 // Whether we should fill the right (or left) side of this contour.
494 bool fill_right_side_;
495
496 DISALLOW_COPY_AND_ASSIGN(Contour);
497 };
498
499 //----------------------------------------------------------------------
500 // Segment
501 //
502
503 // Definition of Segment::Triangulate(), which must come after
504 // declaration of Contour.
505 void Segment::Triangulate(bool compute_inside_edges,
506 CubicTextureCoords::Result* tex_coords) {
507 DCHECK(kind_ == kCubic);
508 if (triangulator_ == NULL) {
509 triangulator_ = arena_->Alloc<LocalTriangulator>();
510 }
511 triangulator_->Reset();
512 for (int i = 0; i < 4; i++) {
513 LocalTriangulator::Vertex* vertex = triangulator_->get_vertex(i);
514 if (tex_coords) {
515 vertex->Set(get_point(i).fX,
516 get_point(i).fY,
517 tex_coords->coords[i].getX(),
518 tex_coords->coords[i].getY(),
519 tex_coords->coords[i].getZ());
520 } else {
521 vertex->Set(get_point(i).fX,
522 get_point(i).fY,
523 // No texture coordinates yet
524 0, 0, 0);
525 }
526 }
527 triangulator_->Triangulate(compute_inside_edges,
528 contour()->fill_right_side());
529 }
530
531 //----------------------------------------------------------------------
532 // PathProcessor
533 //
534
535 PathProcessor::PathProcessor()
536 : arena_(new Arena()),
537 should_delete_arena_(true),
538 verbose_logging_(false) {
539 }
540
541 PathProcessor::PathProcessor(Arena* arena)
542 : arena_(arena),
543 should_delete_arena_(false),
544 verbose_logging_(false) {
545 }
546
547 PathProcessor::~PathProcessor() {
548 if (should_delete_arena_) {
549 delete arena_;
550 }
551 }
552
553 void PathProcessor::Process(const SkPath& path, PathCache* cache) {
554 BuildContours(path);
555
556 // Run plane-sweep algorithm to determine overlaps of control point
557 // curves and subdivide curves appropriately.
558 SubdivideCurves();
559
560 // Determine orientations of countours. Based on orientation and the
561 // number of curve crossings at a random point on the contour,
562 // determine whether to fill the left or right side of the contour.
563 DetermineSidesToFill();
564
565 // Classify curves, compute texture coordinates and subdivide as
566 // necessary to eliminate rendering artifacts. Do the final
567 // triangulation of the curve segments, determining the path along
568 // the interior of the shape.
569 for (std::vector<Contour*>::iterator iter = contours_.begin();
570 iter != contours_.end();
571 iter++) {
572 Contour* cur = *iter;
573 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) {
574 if (seg->kind() == Segment::kCubic) {
575 CubicClassifier::Result classification =
576 CubicClassifier::Classify(seg->get_point(0).fX,
577 seg->get_point(0).fY,
578 seg->get_point(1).fX,
579 seg->get_point(1).fY,
580 seg->get_point(2).fX,
581 seg->get_point(2).fY,
582 seg->get_point(3).fX,
583 seg->get_point(3).fY);
584 DLOG_IF(INFO, verbose_logging_) << "Classification:"
585 << classification.curve_type();
586 CubicTextureCoords::Result tex_coords;
587 CubicTextureCoords::Compute(classification,
588 cur->fill_right_side(),
589 &tex_coords);
590 if (tex_coords.has_rendering_artifact) {
591 // TODO(kbr): split at the subdivision parameter value
592 cur->Subdivide(seg);
593 // Next iteration will handle the newly subdivided halves
594 } else {
595 if (!tex_coords.is_line_or_point) {
596 seg->Triangulate(true, &tex_coords);
597 for (int i = 0; i < seg->num_triangles(); i++) {
598 LocalTriangulator::Triangle* triangle =
599 seg->get_triangle(i);
600 for (int j = 0; j < 3; j++) {
601 LocalTriangulator::Vertex* vert =
602 triangle->get_vertex(j);
603 cache->AddVertex(vert->x(),
604 vert->y(),
605 vert->k(),
606 vert->l(),
607 vert->m());
608 }
609 }
610 #ifdef O3D_CORE_CROSS_GPU2D_PATH_CACHE_DEBUG_INTERIOR_EDGES
611 // Show the end user the interior edges as well
612 for (int i = 1; i < seg->num_interior_vertices(); i++) {
613 SkPoint vert = seg->get_interior_vertex(i);
614 // Duplicate previous vertex to be able to draw GL_LINES
615 SkPoint prev = seg->get_interior_vertex(i - 1);
616 cache->AddInteriorEdgeVertex(prev.fX, prev.fY);
617 cache->AddInteriorEdgeVertex(vert.fX, vert.fY);
618 }
619 #endif // O3D_CORE_CROSS_GPU2D_PATH_CACHE_DEBUG_INTERIOR_EDGES
620 }
621 }
622 }
623 }
624 }
625
626 // Run the interior paths through a tessellation algorithm
627 // supporting multiple contours.
628 TessellateInterior(cache);
629 }
630
631 void PathProcessor::BuildContours(const SkPath& path) {
632 // Clear out the contours
633 contours_.clear();
634 SkPath::Iter iter(path, false);
635 SkPoint pts[4];
636 SkPath::Verb verb;
637 Contour* contour = NULL;
638 SkPoint cur_pt;
639 SkPoint move_to_pt;
640 do {
641 verb = iter.next(pts);
642 if (verb != SkPath::kMove_Verb) {
643 if (contour == NULL) {
644 contour = arena_->Alloc<Contour>();
645 contours_.push_back(contour);
646 }
647 }
648 switch (verb) {
649 case SkPath::kMove_Verb: {
650 contour = arena_->Alloc<Contour>();
651 contours_.push_back(contour);
652 cur_pt = pts[0];
653 move_to_pt = pts[0];
654 DLOG_IF(INFO, verbose_logging_) << "MoveTo (" << pts[0].fX
655 << ", " << pts[0].fY << ")";
656 break;
657 }
658 case SkPath::kLine_Verb: {
659 Segment* segment = arena_->Alloc<Segment>();
660 if (iter.isCloseLine()) {
661 segment->Setup(arena_, contour, cur_pt, pts[1]);
662 DLOG_IF(INFO, verbose_logging_) << "CloseLineTo (" << cur_pt.fX
663 << ", " << cur_pt.fY
664 << "), (" << pts[1].fX
665 << ", " << pts[1].fY << ")";
666 contour->Add(segment);
667 contour = NULL;
668 } else {
669 segment->Setup(arena_, contour, pts[0], pts[1]);
670 DLOG_IF(INFO, verbose_logging_) << "LineTo (" << pts[0].fX
671 << ", " << pts[0].fY
672 << "), (" << pts[1].fX
673 << ", " << pts[1].fY << ")";
674 contour->Add(segment);
675 cur_pt = pts[1];
676 }
677 break;
678 }
679 case SkPath::kQuad_Verb: {
680 // Need to degree elevate the quadratic into a cubic
681 SkPoint cubic[4];
682 SkConvertQuadToCubic(pts, cubic);
683 Segment* segment = arena_->Alloc<Segment>();
684 segment->Setup(arena_, contour,
685 cubic[0], cubic[1], cubic[2], cubic[3]);
686 DLOG_IF(INFO, verbose_logging_) << "Quad->CubicTo (" << cubic[0].fX
687 << ", " << cubic[0].fY
688 << "), (" << cubic[1].fX
689 << ", " << cubic[1].fY
690 << "), (" << cubic[2].fX
691 << ", " << cubic[2].fY
692 << "), (" << cubic[3].fX
693 << ", " << cubic[3].fY << ")";
694 contour->Add(segment);
695 cur_pt = cubic[3];
696 break;
697 }
698 case SkPath::kCubic_Verb: {
699 Segment* segment = arena_->Alloc<Segment>();
700 segment->Setup(arena_, contour, pts[0], pts[1], pts[2], pts[3]);
701 DLOG_IF(INFO, verbose_logging_) << "CubicTo (" << pts[0].fX
702 << ", " << pts[0].fY
703 << "), (" << pts[1].fX
704 << ", " << pts[1].fY
705 << "), (" << pts[2].fX
706 << ", " << pts[2].fY
707 << "), (" << pts[3].fX
708 << ", " << pts[3].fY << ")";
709 contour->Add(segment);
710 cur_pt = pts[3];
711 break;
712 }
713 case SkPath::kClose_Verb: {
714 Segment* segment = arena_->Alloc<Segment>();
715 segment->Setup(arena_, contour, cur_pt, move_to_pt);
716 DLOG_IF(INFO, verbose_logging_) << "Close (" << cur_pt.fX
717 << ", " << cur_pt.fY
718 << ") -> (" << move_to_pt.fX
719 << ", " << move_to_pt.fY << ")";
720 contour->Add(segment);
721 contour = NULL;
722 }
723 default:
724 break;
725 }
726 } while (verb != SkPath::kDone_Verb);
727 }
728
729 std::vector<Segment*> PathProcessor::AllSegmentsOverlappingY(float y) {
730 std::vector<Segment*> res;
731 for (std::vector<Contour*>::iterator iter = contours_.begin();
732 iter != contours_.end();
733 iter++) {
734 Contour* cur = *iter;
735 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) {
736 const BBox& bbox = seg->bbox();
737 if (bbox.min_y() <= y && y <= bbox.max_y()) {
738 res.push_back(seg);
739 }
740 }
741 }
742 return res;
743 }
744
745 // Uncomment this to debug the orientation computation
746 // #define O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION
747
748 void PathProcessor::DetermineSidesToFill() {
749 // Loop and Blinn's algorithm can only easily emulate the even/odd
750 // fill rule, and only for non-intersecting curves. We can determine
751 // which side of each curve segment to fill based on its
752 // clockwise/counterclockwise orientation and how many other
753 // contours surround it.
754
755 // To optimize the query of all curve segments intersecting a
756 // horizontal line going to x=+infinity, we build up an interval
757 // tree whose keys are the y extents of the segments.
758 IntervalTree<float, Segment*> tree(arena_);
759 typedef IntervalTree<float, Segment*>::IntervalType IntervalType;
760
761 for (std::vector<Contour*>::iterator iter = contours_.begin();
762 iter != contours_.end();
763 iter++) {
764 Contour* cur = *iter;
765 DetermineOrientation(cur);
766 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) {
767 const BBox& bbox = seg->bbox();
768 tree.Insert(tree.MakeInterval(bbox.min_y(), bbox.max_y(), seg));
769 }
770 }
771
772 // Now iterate through the contours and pick a random segment (in
773 // this case we use the first) and a random point on that segment.
774 // Find all segments from other contours which intersect this one
775 // and count the number of crossings a horizontal line to
776 // x=+infinity makes with those contours. This combined with the
777 // orientation of the curve tells us which side to fill -- again,
778 // assuming an even/odd fill rule, which is all we can easily
779 // handle.
780 for (std::vector<Contour*>::iterator iter = contours_.begin();
781 iter != contours_.end();
782 iter++) {
783 Contour* cur = *iter;
784 Segment* seg = cur->begin();
785
786 int num_crossings = 0;
787
788 // We use a zero-sized vertical interval for the query
789 std::vector<IntervalType> overlaps =
790 tree.AllOverlaps(IntervalType(SkScalarToFloat(seg->get_point(0).fY),
791 SkScalarToFloat(seg->get_point(0).fY),
792 NULL));
793 #if !defined(O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION)
794 for (std::vector<IntervalType>::iterator iter = overlaps.begin();
795 iter != overlaps.end();
796 iter++) {
797 const IntervalType& interval = *iter;
798 Segment* query_seg = interval.data();
799 // Ignore segments coming from the same contour
800 if (query_seg->contour() != cur) {
801 num_crossings += query_seg->NumCrossingsForXRay(seg->get_point(0));
802 }
803 }
804 #endif // !defined(O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION)
805
806 #ifdef O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION
807 // For debugging
808 std::vector<Segment*> slow_overlaps =
809 AllSegmentsOverlappingY(seg->get_point(0).fY);
810 DCHECK(overlaps.size() == slow_overlaps.size());
811 for (std::vector<Segment*>::iterator iter = slow_overlaps.begin();
812 iter != slow_overlaps.end();
813 iter++) {
814 Segment* query_seg = *iter;
815 // Ignore segments coming from the same contour
816 if (query_seg->contour() != cur) {
817 num_crossings += query_seg->NumCrossingsForXRay(seg->get_point(0));
818 }
819 }
820 #endif // O3D_CORE_CROSS_GPU2D_PATH_PROCESSOR_DEBUG_ORIENTATION
821
822 if (cur->ccw()) {
823 if (num_crossings & 1) {
824 cur->set_fill_right_side(true);
825 } else {
826 cur->set_fill_right_side(false);
827 }
828 } else {
829 if (num_crossings & 1) {
830 cur->set_fill_right_side(false);
831 } else {
832 cur->set_fill_right_side(true);
833 }
834 }
835 }
836 }
837
838 void PathProcessor::DetermineOrientation(Contour* contour) {
839 // Determine signed area of the polygon represented by the points
840 // along the segments. Consider this an approximation to the true
841 // orientation of the polygon; it probably won't handle
842 // self-intersecting curves correctly.
843 //
844 // There is also a pretty basic assumption here that the contour is
845 // closed.
846 float signed_area = 0;
847 for (Segment* seg = contour->begin();
848 seg != contour->end();
849 seg = seg->next()) {
850 int limit = (seg->kind() == Segment::kCubic) ? 4 : 2;
851 for (int i = 1; i < limit; i++) {
852 const SkPoint& prev_point = seg->get_point(i - 1);
853 const SkPoint& point = seg->get_point(i);
854 float cur_area = prev_point.fX * point.fY - prev_point.fY * point.fX;
855 DLOG_IF(INFO, verbose_logging_) << "Adding to signed area ("
856 << prev_point.fX << ", "
857 << prev_point.fY << ") -> ("
858 << point.fX << ", "
859 << point.fY << ") = "
860 << cur_area;
861 signed_area += cur_area;
862 }
863 }
864
865 if (signed_area > 0)
866 contour->set_ccw(true);
867 else
868 contour->set_ccw(false);
869 }
870
871 //----------------------------------------------------------------------
872 // Classes and typedefs needed for curve subdivision.
873 // Unfortunately it appears we can't scope these within the
874 // SubdivideCurves() method itself, because templates then fail to
875 // instantiate.
876
877 // The user data which is placed in the IntervalTree.
878 struct SweepData {
879 SweepData()
880 : triangle(NULL),
881 segment(NULL) {
882 }
883
884 // The triangle this interval is associated with
885 LocalTriangulator::Triangle* triangle;
886 // The segment the triangle is associated with
887 Segment* segment;
888 };
889
890 typedef IntervalTree<float, SweepData*> SweepTree;
891 typedef SweepTree::IntervalType SweepInterval;
892
893 // The entry / exit events which occur at the minimum and maximum x
894 // coordinates of the control point triangles' bounding boxes.
895 //
896 // Note that this class requires its copy constructor and assignment
897 // operator since it needs to be stored in a std::vector.
898 class SweepEvent {
899 public:
900 SweepEvent()
901 : x_(0),
902 entry_(false),
903 interval_(0, 0, NULL) {
904 }
905
906 // Initializes the SweepEvent.
907 void Setup(float x, bool entry, SweepInterval interval) {
908 x_ = x;
909 entry_ = entry;
910 interval_ = interval;
911 }
912
913 float x() const { return x_; }
914 bool entry() const { return entry_; }
915 const SweepInterval& interval() const { return interval_; }
916
917 bool operator<(const SweepEvent& other) const {
918 return x_ < other.x_;
919 }
920
921 private:
922 float x_;
923 bool entry_;
924 SweepInterval interval_;
925 };
926
927 namespace {
928
929 bool TrianglesOverlap(LocalTriangulator::Triangle* t0,
930 LocalTriangulator::Triangle* t1) {
931 LocalTriangulator::Vertex* t0v0 = t0->get_vertex(0);
932 LocalTriangulator::Vertex* t0v1 = t0->get_vertex(1);
933 LocalTriangulator::Vertex* t0v2 = t0->get_vertex(2);
934 LocalTriangulator::Vertex* t1v0 = t1->get_vertex(0);
935 LocalTriangulator::Vertex* t1v1 = t1->get_vertex(1);
936 LocalTriangulator::Vertex* t1v2 = t1->get_vertex(2);
937 return cubic::TrianglesOverlap(t0v0->x(), t0v0->y(),
938 t0v1->x(), t0v1->y(),
939 t0v2->x(), t0v2->y(),
940 t1v0->x(), t1v0->y(),
941 t1v1->x(), t1v1->y(),
942 t1v2->x(), t1v2->y());
943 }
944
945 } // anonymous namespace
946
947 void PathProcessor::SubdivideCurves() {
948 // We need to determine all overlaps of all control point triangles
949 // (from different segments, not the same segment) and, if any
950 // exist, subdivide the associated curves.
951 //
952 // The plane-sweep algorithm determines all overlaps of a set of
953 // rectangles in the 2D plane. Our problem maps very well to this
954 // algorithm and significantly reduces the complexity compared to a
955 // naive implementation.
956 //
957 // Each bounding box of a control point triangle is converted into
958 // an "entry" event at its smallest X coordinate and an "exit" event
959 // at its largest X coordinate. Each event has an associated
960 // one-dimensional interval representing the Y span of the bounding
961 // box. We sort these events by increasing X coordinate. We then
962 // iterate through them. For each entry event we add the interval to
963 // a side interval tree, and query this tree for overlapping
964 // intervals. Any overlapping interval corresponds to an overlapping
965 // bounding box. For each exit event we remove the associated
966 // interval from the interval tree.
967
968 std::vector<Segment*> cur_segments;
969 std::vector<Segment*> next_segments;
970
971 // Start things off by considering all of the segments
972 for (std::vector<Contour*>::iterator iter = contours_.begin();
973 iter != contours_.end();
974 iter++) {
975 Contour* cur = *iter;
976 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) {
977 if (seg->kind() == Segment::kCubic) {
978 seg->Triangulate(false, NULL);
979 cur_segments.push_back(seg);
980 }
981 }
982 }
983
984 // Subdivide curves at most this many times
985 const int kMaxIter = 5;
986 std::vector<SweepInterval> overlaps;
987
988 for (int cur_iter = 0; cur_iter < kMaxIter; ++cur_iter) {
989 if (cur_segments.size() == 0) {
990 // Done
991 break;
992 }
993
994 std::vector<SweepEvent> events;
995 SweepTree tree(arena_);
996 for (std::vector<Segment*>::iterator iter = cur_segments.begin();
997 iter != cur_segments.end();
998 iter++) {
999 Segment* seg = *iter;
1000 if (seg->kind() == Segment::kCubic) {
1001 for (int i = 0; i < seg->num_triangles(); i++) {
1002 LocalTriangulator::Triangle* triangle = seg->get_triangle(i);
1003 BBox bbox;
1004 bbox.Setup(triangle);
1005 // Ignore zero-width triangles to avoid issues with
1006 // coincident entry and exit events for the same triangle
1007 if (bbox.max_x() > bbox.min_x()) {
1008 SweepData* data = arena_->Alloc<SweepData>();
1009 data->triangle = triangle;
1010 data->segment = seg;
1011 SweepInterval interval =
1012 tree.MakeInterval(bbox.min_y(), bbox.max_y(), data);
1013 // Add entry and exit events
1014 SweepEvent event;
1015 event.Setup(bbox.min_x(), true, interval);
1016 events.push_back(event);
1017 event.Setup(bbox.max_x(), false, interval);
1018 events.push_back(event);
1019 }
1020 }
1021 }
1022 }
1023
1024 // Sort events by increasing X coordinate
1025 std::sort(events.begin(), events.end());
1026
1027 // Now iterate through the events
1028 for (std::vector<SweepEvent>::iterator iter = events.begin();
1029 iter != events.end();
1030 iter++) {
1031 SweepEvent event = *iter;
1032 if (event.entry()) {
1033 // Add this interval into the tree
1034 tree.Insert(event.interval());
1035 // See whether the associated segment has been subdivided yet
1036 if (!event.interval().data()->segment->marked_for_subdivision()) {
1037 // Query the tree
1038 overlaps.clear();
1039 tree.AllOverlaps(event.interval(), overlaps);
1040 // Now see exactly which triangles overlap this one
1041 for (std::vector<SweepInterval>::iterator iter = overlaps.begin();
1042 iter != overlaps.end();
1043 iter++) {
1044 SweepInterval overlap = *iter;
1045 // Only pay attention to overlaps from a different Segment
1046 if (event.interval().data()->segment != overlap.data()->segment) {
1047 // See whether the triangles actually overlap
1048 if (TrianglesOverlap(event.interval().data()->triangle,
1049 overlap.data()->triangle)) {
1050 // Actually subdivide the segments.
1051 // Each one might already have been subdivided.
1052 Segment* seg = event.interval().data()->segment;
1053 ConditionallySubdivide(seg, &next_segments);
1054 seg = overlap.data()->segment;
1055 ConditionallySubdivide(seg, &next_segments);
1056 }
1057 }
1058 }
1059 }
1060 } else {
1061 // Remove this interval from the tree
1062 tree.Delete(event.interval());
1063 }
1064 }
1065
1066 cur_segments = next_segments;
1067 next_segments.clear();
1068 }
1069 }
1070
1071 void PathProcessor::ConditionallySubdivide(
1072 Segment* seg, std::vector<Segment*>* next_segments) {
1073 if (!seg->marked_for_subdivision()) {
1074 seg->set_marked_for_subdivision(true);
1075 Segment* next = seg->contour()->Subdivide(seg);
1076 // Triangulate the newly subdivided segments.
1077 next->Triangulate(false, NULL);
1078 next->next()->Triangulate(false, NULL);
1079 // Add them for the next iteration.
1080 next_segments->push_back(next);
1081 next_segments->push_back(next->next());
1082 }
1083 }
1084
1085 void PathProcessor::SubdivideCurvesSlow() {
1086 // Alternate, significantly slower algorithm for curve subdivision
1087 // for use in debugging.
1088 std::vector<Segment*> cur_segments;
1089 std::vector<Segment*> next_segments;
1090
1091 // Start things off by considering all of the segments
1092 for (std::vector<Contour*>::iterator iter = contours_.begin();
1093 iter != contours_.end();
1094 iter++) {
1095 Contour* cur = *iter;
1096 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) {
1097 if (seg->kind() == Segment::kCubic) {
1098 seg->Triangulate(false, NULL);
1099 cur_segments.push_back(seg);
1100 }
1101 }
1102 }
1103
1104 // Subdivide curves at most this many times
1105 const int kMaxIter = 5;
1106
1107 for (int cur_iter = 0; cur_iter < kMaxIter; ++cur_iter) {
1108 if (cur_segments.size() == 0) {
1109 // Done
1110 break;
1111 }
1112
1113 for (std::vector<Segment*>::iterator iter = cur_segments.begin();
1114 iter != cur_segments.end();
1115 iter++) {
1116 Segment* seg = *iter;
1117 if (seg->kind() == Segment::kCubic) {
1118 for (std::vector<Segment*>::iterator iter2 = cur_segments.begin();
1119 iter2 != cur_segments.end();
1120 iter2++) {
1121 Segment* seg2 = *iter2;
1122 if (seg2->kind() == Segment::kCubic &&
1123 seg != seg2) {
1124 for (int i = 0; i < seg->num_triangles(); i++) {
1125 LocalTriangulator::Triangle* triangle = seg->get_triangle(i);
1126 for (int j = 0; j < seg2->num_triangles(); j++) {
1127 LocalTriangulator::Triangle* triangle2 = seg2->get_triangle(j);
1128 if (TrianglesOverlap(triangle, triangle2)) {
1129 ConditionallySubdivide(seg, &next_segments);
1130 ConditionallySubdivide(seg2, &next_segments);
1131 }
1132 }
1133 }
1134 }
1135 }
1136 }
1137 }
1138
1139 cur_segments = next_segments;
1140 next_segments.clear();
1141 }
1142 }
1143
1144 //----------------------------------------------------------------------
1145 // Structures and callbacks for tessellation of the interior region of
1146 // the contours.
1147
1148 // The user data for the GLU tessellator.
1149 struct TessellationState {
1150 PathCache* cache;
1151 std::vector<void*> allocated_pointers;
1152 };
1153
1154 namespace {
1155
1156 void VertexCallback(void* vertex_data, void* data) {
1157 TessellationState* state = static_cast<TessellationState*>(data);
1158 PathCache* cache = state->cache;
1159 GLdouble* location = static_cast<GLdouble*>(vertex_data);
1160 cache->AddInteriorVertex(static_cast<float>(location[0]),
1161 static_cast<float>(location[1]));
1162 }
1163
1164 void CombineCallback(GLdouble coords[3], void* vertex_data[4],
1165 GLfloat weight[4], void** out_data,
1166 void* polygon_data) {
1167 TessellationState* state = static_cast<TessellationState*>(polygon_data);
1168 GLdouble* out_vertex = static_cast<GLdouble*>(malloc(3 * sizeof(GLdouble)));
1169 state->allocated_pointers.push_back(out_vertex);
1170 out_vertex[0] = coords[0];
1171 out_vertex[1] = coords[1];
1172 out_vertex[2] = coords[2];
1173 *out_data = out_vertex;
1174 }
1175
1176 void EdgeFlagCallback(GLboolean flag) {
1177 // No-op just to prevent triangle strips and fans from being passed
1178 // to us
1179 }
1180
1181 } // anonymous namespace
1182
1183 void PathProcessor::TessellateInterior(PathCache* cache) {
1184 // Because the GLU tessellator requires its input in
1185 // double-precision format, we need to make a separate copy of the
1186 // data.
1187 std::vector<GLdouble> vertex_data;
1188 std::vector<size_t> contour_endings;
1189 // For avoiding adding coincident vertices.
1190 float cur_x = 0, cur_y = 0;
1191 for (std::vector<Contour*>::iterator iter = contours_.begin();
1192 iter != contours_.end();
1193 iter++) {
1194 Contour* cur = *iter;
1195 bool first = true;
1196 for (Segment* seg = cur->begin(); seg != cur->end(); seg = seg->next()) {
1197 int num_interior_vertices = seg->num_interior_vertices();
1198 for (int i = 0; i < num_interior_vertices - 1; i++) {
1199 SkPoint point = seg->get_interior_vertex(i);
1200 if (first) {
1201 first = false;
1202 vertex_data.push_back(point.fX);
1203 vertex_data.push_back(point.fY);
1204 vertex_data.push_back(0);
1205 cur_x = point.fX;
1206 cur_y = point.fY;
1207 } else if (point.fX != cur_x || point.fY != cur_y) {
1208 vertex_data.push_back(point.fX);
1209 vertex_data.push_back(point.fY);
1210 vertex_data.push_back(0);
1211 cur_x = point.fX;
1212 cur_y = point.fY;
1213 }
1214 }
1215 }
1216 contour_endings.push_back(vertex_data.size());
1217 }
1218 // Now that we have all of the vertex data in a stable location in
1219 // memory, call the tessellator.
1220 GLUtesselator* tess = internal_gluNewTess();
1221 TessellationState state;
1222 state.cache = cache;
1223 internal_gluTessCallback(tess, GLU_TESS_VERTEX_DATA,
1224 reinterpret_cast<GLvoid (*)()>(VertexCallback));
1225 internal_gluTessCallback(tess, GLU_TESS_COMBINE_DATA,
1226 reinterpret_cast<GLvoid (*)()>(CombineCallback));
1227 internal_gluTessCallback(tess, GLU_TESS_EDGE_FLAG,
1228 reinterpret_cast<GLvoid (*)()>(EdgeFlagCallback));
1229 internal_gluTessBeginPolygon(tess, &state);
1230 internal_gluTessBeginContour(tess);
1231 GLdouble* base = &vertex_data.front();
1232 int contour_idx = 0;
1233 for (size_t i = 0; i < vertex_data.size(); i += 3) {
1234 if (i == contour_endings[contour_idx]) {
1235 internal_gluTessEndContour(tess);
1236 internal_gluTessBeginContour(tess);
1237 ++contour_idx;
1238 }
1239 internal_gluTessVertex(tess, &base[i], &base[i]);
1240 }
1241 internal_gluTessEndContour(tess);
1242 internal_gluTessEndPolygon(tess);
1243 for (size_t i = 0; i < state.allocated_pointers.size(); i++) {
1244 free(state.allocated_pointers[i]);
1245 }
1246 internal_gluDeleteTess(tess);
1247 }
1248
1249 } // namespace gpu2d
1250 } // namespace o3d
1251
OLDNEW
« no previous file with comments | « core/cross/gpu2d/path_processor.h ('k') | core/cross/primitive.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698