aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--include/core/SkPath.h10
-rw-r--r--src/core/SkPath.cpp21
-rw-r--r--tests/PathTest.cpp21
3 files changed, 36 insertions, 16 deletions
diff --git a/include/core/SkPath.h b/include/core/SkPath.h
index d4b79cda04..be15a63c89 100644
--- a/include/core/SkPath.h
+++ b/include/core/SkPath.h
@@ -40,11 +40,10 @@ public:
SK_DECLARE_INST_COUNT_ROOT(SkPath);
SkPath();
- SkPath(const SkPath&);
+ SkPath(const SkPath&); // Copies fGenerationID on Android.
~SkPath();
- SkPath& operator=(const SkPath&);
-
+ SkPath& operator=(const SkPath&); // Increments fGenerationID on Android.
friend SK_API bool operator==(const SkPath&, const SkPath&);
friend bool operator!=(const SkPath& a, const SkPath& b) {
return !(a == b);
@@ -167,14 +166,12 @@ public:
/** Clear any lines and curves from the path, making it empty. This frees up
internal storage associated with those segments.
- This does NOT change the fill-type setting nor isConvex
*/
void reset();
/** Similar to reset(), in that all lines and curves are removed from the
path. However, any internal storage for those lines/curves is retained,
making reuse of the path potentially faster.
- This does NOT change the fill-type setting nor isConvex
*/
void rewind();
@@ -965,8 +962,7 @@ private:
/** Sets all fields other than fPathRef to the values in 'that'.
* Assumes the caller has already set fPathRef.
- * On Android increments fGenerationID without copying it.
- * On Android sets fSourcePath to NULL.
+ * Doesn't change fGenerationID or fSourcePath on Android.
*/
void copyFields(const SkPath& that);
diff --git a/src/core/SkPath.cpp b/src/core/SkPath.cpp
index 875be9801e..f5b8bd5e79 100644
--- a/src/core/SkPath.cpp
+++ b/src/core/SkPath.cpp
@@ -185,12 +185,12 @@ void SkPath::resetFields() {
}
SkPath::SkPath(const SkPath& that)
- : fPathRef(SkRef(that.fPathRef.get()))
+ : fPathRef(SkRef(that.fPathRef.get())) {
+ this->copyFields(that);
#ifdef SK_BUILD_FOR_ANDROID
- , fGenerationID(0)
+ fGenerationID = that.fGenerationID;
+ fSourcePath = NULL; // TODO(mtklein): follow up with Android: do we want to copy this too?
#endif
-{
- this->copyFields(that);
SkDEBUGCODE(that.validate();)
}
@@ -204,6 +204,10 @@ SkPath& SkPath::operator=(const SkPath& that) {
if (this != &that) {
fPathRef.reset(SkRef(that.fPathRef.get()));
this->copyFields(that);
+#ifdef SK_BUILD_FOR_ANDROID
+ GEN_ID_INC; // Similar to swap, we can't just copy this or it could go back in time.
+ fSourcePath = NULL; // TODO(mtklein): follow up with Android: do we want to copy this too?
+#endif
}
SkDEBUGCODE(this->validate();)
return *this;
@@ -220,10 +224,6 @@ void SkPath::copyFields(const SkPath& that) {
fDirection = that.fDirection;
fIsFinite = that.fIsFinite;
fIsOval = that.fIsOval;
-#ifdef SK_BUILD_FOR_ANDROID
- GEN_ID_INC;
- fSourcePath = NULL;
-#endif
}
SK_API bool operator==(const SkPath& a, const SkPath& b) {
@@ -253,8 +253,13 @@ void SkPath::swap(SkPath& that) {
SkTSwap<uint8_t>(fDirection, that.fDirection);
SkTSwap<SkBool8>(fIsFinite, that.fIsFinite);
SkTSwap<SkBool8>(fIsOval, that.fIsOval);
+#ifdef SK_BUILD_FOR_ANDROID
+ // It doesn't really make sense to swap the generation IDs here, because they might go
+ // backwards. To be safe we increment both to mark them both as changed.
GEN_ID_INC;
GEN_ID_PTR_INC(&that);
+ SkTSwap<const SkPath*>(fSourcePath, that.fSourcePath);
+#endif
}
}
diff --git a/tests/PathTest.cpp b/tests/PathTest.cpp
index e33c912062..e698c7cda5 100644
--- a/tests/PathTest.cpp
+++ b/tests/PathTest.cpp
@@ -15,8 +15,9 @@
#include "SkRandom.h"
#include "SkReader32.h"
#include "SkSize.h"
-#include "SkWriter32.h"
#include "SkSurface.h"
+#include "SkTypes.h"
+#include "SkWriter32.h"
#if defined(WIN32)
#define SUPPRESS_VISIBILITY_WARNING
@@ -31,6 +32,23 @@ static SkSurface* new_surface(int w, int h) {
return SkSurface::NewRaster(info);
}
+static void test_android_specific_behavior(skiatest::Reporter* reporter) {
+#ifdef SK_BUILD_FOR_ANDROID
+ // Copy constructor should preserve generation ID, but assignment shouldn't.
+ SkPath original;
+ original.moveTo(0, 0);
+ original.lineTo(1, 1);
+ REPORTER_ASSERT(reporter, original.getGenerationID() > 0);
+
+ const SkPath copy(original);
+ REPORTER_ASSERT(reporter, copy.getGenerationID() == original.getGenerationID());
+
+ SkPath assign;
+ assign = original;
+ REPORTER_ASSERT(reporter, assign.getGenerationID() != original.getGenerationID());
+#endif
+}
+
// This used to assert in the debug build, as the edges did not all line-up.
static void test_bad_cubic_crbug234190() {
SkPath path;
@@ -2450,6 +2468,7 @@ static void TestPath(skiatest::Reporter* reporter) {
test_crbug_170666();
test_bad_cubic_crbug229478();
test_bad_cubic_crbug234190();
+ test_android_specific_behavior(reporter);
}
#include "TestClassDef.h"