summaryrefslogtreecommitdiffstats
path: root/o3d/core
diff options
context:
space:
mode:
authorkbr@chromium.org <kbr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-02-18 22:27:01 +0000
committerkbr@chromium.org <kbr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-02-18 22:27:01 +0000
commitcef17aeea0413bfdcbd404b837aac6a054024d95 (patch)
treede0116ccfd4d777662abdefd6f91354e9a44718d /o3d/core
parente585f48a189f0e6558c6adc583738c6271980f98 (diff)
downloadchromium_src-cef17aeea0413bfdcbd404b837aac6a054024d95.zip
chromium_src-cef17aeea0413bfdcbd404b837aac6a054024d95.tar.gz
chromium_src-cef17aeea0413bfdcbd404b837aac6a054024d95.tar.bz2
Added computation of three-dimensional texture coordinates for the
control points of cubic curves. It's difficult if not impossible to write a unit test for this part of the algorithm, since the texture coordinates aren't precisely defined for any particular inputs. The forthcoming system tests will exercise these code paths. BUG=none TEST=none Review URL: http://codereview.chromium.org/647041 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@39386 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'o3d/core')
-rw-r--r--o3d/core/core.gyp2
-rw-r--r--o3d/core/cross/gpu2d/cubic_texture_coords.cc193
-rw-r--r--o3d/core/cross/gpu2d/cubic_texture_coords.h94
3 files changed, 289 insertions, 0 deletions
diff --git a/o3d/core/core.gyp b/o3d/core/core.gyp
index c979bae..33e8740 100644
--- a/o3d/core/core.gyp
+++ b/o3d/core/core.gyp
@@ -273,6 +273,8 @@
'cross/gpu2d/cubic_classifier.cc',
'cross/gpu2d/cubic_classifier.h',
'cross/gpu2d/cubic_math_utils.h',
+ 'cross/gpu2d/cubic_texture_coords.cc',
+ 'cross/gpu2d/cubic_texture_coords.h',
'cross/gpu2d/interval_tree.h',
'cross/gpu2d/red_black_tree.h',
],
diff --git a/o3d/core/cross/gpu2d/cubic_texture_coords.cc b/o3d/core/cross/gpu2d/cubic_texture_coords.cc
new file mode 100644
index 0000000..f797b87
--- /dev/null
+++ b/o3d/core/cross/gpu2d/cubic_texture_coords.cc
@@ -0,0 +1,193 @@
+/*
+ * Copyright 2010, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <math.h>
+
+#include "base/logging.h"
+#include "core/cross/gpu2d/cubic_math_utils.h"
+#include "core/cross/gpu2d/cubic_texture_coords.h"
+
+namespace o3d {
+namespace gpu2d {
+
+void CubicTextureCoords::Compute(const CubicClassifier::Result& c,
+ bool fill_right_side,
+ CubicTextureCoords::Result* result) {
+ // Loop and Blinn's formulation states that the right side of the
+ // curve is defined to be the inside (filled region), but for some
+ // reason it looks like with the default orientation parameters we
+ // are filling the left side of the curve. Regardless, because we
+ // can receive arbitrarily oriented curves as input, we might have
+ // to reverse the orientation of the cubic texture coordinates even
+ // in cases where the paper doen't say it is necessary.
+ bool reverse_orientation = false;
+ const float kOneThird = 1.0f / 3.0f;
+ const float kTwoThirds = 2.0f / 3.0f;
+ CubicClassifier::CurveType curve_type = c.curve_type();
+
+ result->is_line_or_point = false;
+ result->has_rendering_artifact = false;
+ result->subdivision_parameter_value = 0.0f;
+
+ switch (curve_type) {
+ case CubicClassifier::kSerpentine: {
+ float t1 = sqrtf(9.0f * c.d2() * c.d2() - 12 * c.d1() * c.d3());
+ float ls = 3.0f * c.d2() - t1;
+ float lt = 6.0f * c.d1();
+ float ms = 3.0f * c.d2() + t1;
+ float mt = lt;
+ float ltmls = lt - ls;
+ float mtmms = mt - ms;
+ result->coords[0] = Vector3(ls * ms,
+ ls * ls * ls,
+ ms * ms * ms);
+ result->coords[1] = Vector3(kOneThird * (3.0f * ls * ms -
+ ls * mt -
+ lt * ms),
+ ls * ls * (ls - lt),
+ ms * ms * (ms - mt));
+ result->coords[2] = Vector3(kOneThird * (lt * (mt - 2.0f * ms) +
+ ls * (3.0f * ms - 2.0f * mt)),
+ ltmls * ltmls * ls,
+ mtmms * mtmms * ms);
+ result->coords[3] = Vector3(ltmls * mtmms,
+ -(ltmls * ltmls * ltmls),
+ -(mtmms * mtmms * mtmms));
+ if (c.d1() < 0.0f)
+ reverse_orientation = true;
+ break;
+ }
+
+ case CubicClassifier::kLoop: {
+ float t1 = sqrtf(4.0f * c.d1() * c.d3() - 3.0f * c.d2() * c.d2());
+ float ls = c.d2() - t1;
+ float lt = 2.0f * c.d1();
+ float ms = c.d2() + t1;
+ float mt = lt;
+
+ // Figure out whether there is a rendering artifact requiring
+ // the curve to be subdivided by the caller.
+ float ql = ls / lt;
+ float qm = ms / mt;
+ if (0.0f < ql && ql < 1.0f) {
+ result->has_rendering_artifact = true;
+ result->subdivision_parameter_value = ql;
+ // TODO(kbr): may require more work?
+ break;
+ }
+
+ if (0.0f < qm && qm < 1.0f) {
+ result->has_rendering_artifact = true;
+ result->subdivision_parameter_value = qm;
+ // TODO(kbr): may require more work?
+ break;
+ }
+
+ float ltmls = lt - ls;
+ float mtmms = mt - ms;
+ result->coords[0] = Vector3(ls * ms,
+ ls * ls * ms,
+ ls * ms * ms);
+ result->coords[1] =
+ Vector3(kOneThird * (-ls * mt - lt * ms + 3.0f * ls * ms),
+ -kOneThird * ls * (ls * (mt - 3.0f * ms) + 2.0f * lt * ms),
+ -kOneThird * ms * (ls * (2.0f * mt - 3.0f * ms) + lt * ms));
+ result->coords[2] =
+ Vector3(kOneThird * (lt * (mt - 2.0f * ms) +
+ ls * (3.0f * ms - 2.0f * mt)),
+ kOneThird * (lt - ls) * (ls * (2.0f * mt - 3.0f * ms) +
+ lt * ms),
+ kOneThird * (mt - ms) * (ls * (mt - 3.0f * ms) +
+ 2.0f * lt * ms));
+ result->coords[3] =
+ Vector3(ltmls * mtmms,
+ -(ltmls * ltmls) * mtmms,
+ -ltmls * mtmms * mtmms);
+ reverse_orientation =
+ ((c.d1() > 0.0f && result->coords[0].getX() < 0.0f) ||
+ (c.d1() < 0.0f && result->coords[0].getX() > 0.0f));
+ break;
+ }
+
+ case CubicClassifier::kCusp: {
+ float ls = c.d3();
+ float lt = 3.0f * c.d2();
+ float lsmlt = ls - lt;
+ result->coords[0] = Vector3(ls,
+ ls * ls * ls,
+ 1.0f);
+ result->coords[1] = Vector3(ls - kOneThird * lt,
+ ls * ls * lsmlt,
+ 1.0f);
+ result->coords[2] = Vector3(ls - kTwoThirds * lt,
+ lsmlt * lsmlt * ls,
+ 1.0f);
+ result->coords[3] = Vector3(lsmlt,
+ lsmlt * lsmlt * lsmlt,
+ 1.0f);
+ break;
+ }
+
+ case CubicClassifier::kQuadratic: {
+ result->coords[0] = Vector3(0.0f, 0.0f, 0.0f);
+ result->coords[1] = Vector3(kOneThird, 0, kOneThird);
+ result->coords[2] = Vector3(kTwoThirds, kOneThird, kTwoThirds);
+ result->coords[3] = Vector3(1.0f, 1.0f, 1.0f);
+ if (c.d3() < 0.0f)
+ reverse_orientation = true;
+ break;
+ }
+
+ case CubicClassifier::kLine:
+ case CubicClassifier::kPoint:
+ result->is_line_or_point = true;
+ break;
+
+ default:
+ NOTREACHED();
+ break;
+ }
+
+ if (fill_right_side) {
+ reverse_orientation = !reverse_orientation;
+ }
+
+ if (reverse_orientation) {
+ for (int i = 0; i < 4; i++) {
+ result->coords[i].setX(-result->coords[i].getX());
+ result->coords[i].setY(-result->coords[i].getY());
+ }
+ }
+}
+
+} // namespace gpu2d
+} // namespace o3d
+
diff --git a/o3d/core/cross/gpu2d/cubic_texture_coords.h b/o3d/core/cross/gpu2d/cubic_texture_coords.h
new file mode 100644
index 0000000..7dab7c5
--- /dev/null
+++ b/o3d/core/cross/gpu2d/cubic_texture_coords.h
@@ -0,0 +1,94 @@
+/*
+ * Copyright 2010, Google Inc.
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following disclaimer
+ * in the documentation and/or other materials provided with the
+ * distribution.
+ * * Neither the name of Google Inc. nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+// Computes three-dimensional texture coordinates for the control
+// points of a cubic curve for rendering via the shader in "Rendering
+// Vector Art on the GPU" by Loop and Blinn, GPU Gems 3, Chapter 25.
+
+#ifndef O3D_CORE_CROSS_GPU2D_CUBIC_TEXTURE_COORDS_H_
+#define O3D_CORE_CROSS_GPU2D_CUBIC_TEXTURE_COORDS_H_
+
+#include "base/basictypes.h"
+#include "core/cross/math_types.h"
+#include "core/cross/gpu2d/cubic_classifier.h"
+
+namespace o3d {
+namespace gpu2d {
+
+// Computes texture coordinates for rendering cubic curves on the GPU.
+class CubicTextureCoords {
+ public:
+ // Container for the cubic texture coordinates and other associated
+ // information.
+ struct Result {
+ Result()
+ : is_line_or_point(false),
+ has_rendering_artifact(false),
+ subdivision_parameter_value(0.0f) {
+ }
+
+ // The 3D texture coordinates that are to be associated with the
+ // four control points of the cubic curve.
+ Vector3 coords[4];
+
+ // Indicates whether the curve is a line or a point, in which case
+ // we do not need to add its triangles to the mesh.
+ bool is_line_or_point;
+
+ // For the loop case, indicates whether a rendering artifact was
+ // detected, in which case the curve needs to be further
+ // subdivided.
+ bool has_rendering_artifact;
+
+ // If a rendering artifact will occur for the given loop curve,
+ // this is the parameter value (0 <= value <= 1) at which the
+ // curve needs to be subdivided to fix the artifact.
+ float subdivision_parameter_value;
+ };
+
+ // Computes the texture coordinates for a cubic curve segment's
+ // control points, given the classification of the curve as well as
+ // an indication of which side is to be filled.
+ static void Compute(const CubicClassifier::Result& classification,
+ bool fill_right_side,
+ Result* result);
+
+ private:
+ // This class does not need to be instantiated.
+ CubicTextureCoords() {}
+ DISALLOW_COPY_AND_ASSIGN(CubicTextureCoords);
+};
+
+} // namespace gpu2d
+} // namespace o3d
+
+#endif // O3D_CORE_CROSS_GPU2D_CUBIC_TEXTURE_COORDS_H_
+