This repository has been archived on 2023-10-09. You can view files and clone it, but cannot push or open issues or pull requests.
Files
blender-archive/source/blender/freestyle/intern/geometry/FitCurve.cpp

584 lines
16 KiB
C++
Raw Normal View History

/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
2008-04-30 15:41:54 +00:00
/** \file
* \ingroup freestyle
* \brief An Algorithm for Automatically Fitting Digitized Curves by Philip J. Schneider,
* \brief from "Graphics Gems", Academic Press, 1990
*/
2008-04-30 15:41:54 +00:00
#include <cmath>
#include <cstdio>
#include <cstdlib> // for malloc and free
2008-04-30 15:41:54 +00:00
#include "FitCurve.h"
using namespace std;
Attempt to fix a potential name conflict between Freestyle and the compositor. A crash in the Freestyle renderer was reported by Ton on IRC with a stack trace below. Note that #2 is in Freestyle, whereas #1 is in the compositor. The problem was observed in a debug build on OS X 10.7 (gcc 4.2, openmp disabled, no llvm). ---------------------------------------------------------------------- Program received signal EXC_BAD_ACCESS, Could not access memory. Reason: 13 at address: 0x0000000000000000 [Switching to process 72386 thread 0xf303] 0x0000000100c129f3 in NodeBase::~NodeBase (this=0x10e501c80) at COM_NodeBase.cpp:43 43 delete (this->m_outputsockets.back()); Current language: auto; currently c++ (gdb) where #0 0x0000000100c129f3 in NodeBase::~NodeBase (this=0x10e501c80) at COM_NodeBase.cpp:43 #1 0x0000000100c29066 in Node::~Node (this=0x10e501c80) at COM_Node.h:49 #2 0x000000010089c273 in NodeShape::~NodeShape (this=0x10e501c80) at NodeShape.cpp:43 #3 0x000000010089910b in NodeGroup::destroy (this=0x10e501da0) at NodeGroup.cpp:61 #4 0x00000001008990cd in NodeGroup::destroy (this=0x10e5014b0) at NodeGroup.cpp:59 #5 0x00000001008990cd in NodeGroup::destroy (this=0x114e18da0) at NodeGroup.cpp:59 #6 0x00000001007e6602 in Controller::ClearRootNode (this=0x114e19640) at Controller.cpp:329 #7 0x00000001007ea52e in Controller::LoadMesh (this=0x114e19640, re=0x10aba4638, srl=0x1140f5258) at Controller.cpp:302 #8 0x00000001008030ad in prepare (re=0x10aba4638, srl=0x1140f5258) at FRS_freestyle.cpp:302 #9 0x000000010080457a in FRS_do_stroke_rendering (re=0x10aba4638, srl=0x1140f5258) at FRS_freestyle.cpp:600 #10 0x00000001006aeb9d in add_freestyle (re=0x10aba4638) at pipeline.c:1584 #11 0x00000001006aceb7 in do_render_3d (re=0x10aba4638) at pipeline.c:1094 #12 0x00000001006ae061 in do_render_fields_blur_3d (re=0x10aba4638) at pipeline.c:1367 #13 0x00000001006afa16 in do_render_composite_fields_blur_3d (re=0x10aba4638) at pipeline.c:1815 #14 0x00000001006b04e4 in do_render_all_options (re=0x10aba4638) at pipeline.c:2021 ---------------------------------------------------------------------- Apparently a name conflict between the two Blender modules is taking place. The present commit hence intends to address it by putting all the Freestyle C++ classes in the namespace 'Freestyle'. This revision will also prevent potential name conflicts with other Blender modules in the future. Special thanks to Lukas Toenne for the help with C++ namespace.
2013-04-09 00:46:49 +00:00
namespace Freestyle {
2008-04-30 15:41:54 +00:00
2020-12-09 16:29:11 +01:00
using BezierCurve = Vector2 *;
2008-04-30 15:41:54 +00:00
/* Forward declarations */
static double *Reparameterize(Vector2 *d, int first, int last, double *u, BezierCurve bezCurve);
static double NewtonRaphsonRootFind(BezierCurve Q, Vector2 P, double u);
static Vector2 BezierII(int degree, Vector2 *V, double t);
static double B0(double u);
static double B1(double u);
static double B2(double u);
static double B3(double u);
static Vector2 ComputeLeftTangent(Vector2 *d, int end);
static double ComputeMaxError(
Vector2 *d, int first, int last, BezierCurve bezCurve, double *u, int *splitPoint);
2008-04-30 15:41:54 +00:00
static double *ChordLengthParameterize(Vector2 *d, int first, int last);
static BezierCurve GenerateBezier(
Vector2 *d, int first, int last, double *uPrime, Vector2 tHat1, Vector2 tHat2);
2008-04-30 15:41:54 +00:00
static Vector2 V2AddII(Vector2 a, Vector2 b);
static Vector2 V2ScaleIII(Vector2 v, double s);
static Vector2 V2SubII(Vector2 a, Vector2 b);
/* returns squared length of input vector */
static double V2SquaredLength(Vector2 *a)
{
return (((*a)[0] * (*a)[0]) + ((*a)[1] * (*a)[1]));
2008-04-30 15:41:54 +00:00
}
2008-04-30 15:41:54 +00:00
/* returns length of input vector */
2018-06-17 17:05:14 +02:00
static double V2Length(Vector2 *a)
2008-04-30 15:41:54 +00:00
{
return (sqrt(V2SquaredLength(a)));
2008-04-30 15:41:54 +00:00
}
static Vector2 *V2Scale(Vector2 *v, double newlen)
2008-04-30 15:41:54 +00:00
{
double len = V2Length(v);
if (len != 0.0) {
(*v)[0] *= newlen / len;
(*v)[1] *= newlen / len;
}
return v;
2008-04-30 15:41:54 +00:00
}
/* return the dot product of vectors a and b */
static double V2Dot(Vector2 *a, Vector2 *b)
2008-04-30 15:41:54 +00:00
{
return (((*a)[0] * (*b)[0]) + ((*a)[1] * (*b)[1]));
2008-04-30 15:41:54 +00:00
}
/* return the distance between two points */
static double V2DistanceBetween2Points(Vector2 *a, Vector2 *b)
2008-04-30 15:41:54 +00:00
{
double dx = (*a)[0] - (*b)[0];
double dy = (*a)[1] - (*b)[1];
return (sqrt((dx * dx) + (dy * dy)));
2008-04-30 15:41:54 +00:00
}
/* return vector sum c = a+b */
static Vector2 *V2Add(Vector2 *a, Vector2 *b, Vector2 *c)
2008-04-30 15:41:54 +00:00
{
(*c)[0] = (*a)[0] + (*b)[0];
(*c)[1] = (*a)[1] + (*b)[1];
return c;
2018-06-17 17:05:14 +02:00
}
2008-04-30 15:41:54 +00:00
/* normalizes the input vector and returns it */
static Vector2 *V2Normalize(Vector2 *v)
2008-04-30 15:41:54 +00:00
{
double len = V2Length(v);
if (len != 0.0) {
(*v)[0] /= len;
(*v)[1] /= len;
}
return v;
2008-04-30 15:41:54 +00:00
}
/* negates the input vector and returns it */
static Vector2 *V2Negate(Vector2 *v)
2008-04-30 15:41:54 +00:00
{
(*v)[0] = -(*v)[0];
(*v)[1] = -(*v)[1];
return v;
2008-04-30 15:41:54 +00:00
}
/* GenerateBezier:
2008-04-30 15:41:54 +00:00
* Use least-squares method to find Bezier control points for region.
* Vector2 *d; Array of digitized points
* int first, last; Indices defining region
* double *uPrime; Parameter values for region
* Vector2 tHat1, tHat2; Unit tangents at endpoints
2008-04-30 15:41:54 +00:00
*/
static BezierCurve GenerateBezier(
Vector2 *d, int first, int last, double *uPrime, Vector2 tHat1, Vector2 tHat2)
2008-04-30 15:41:54 +00:00
{
int i;
Vector2 A[2]; /* rhs for eqn */
int nPts; /* Number of pts in sub-curve */
double C[2][2]; /* Matrix C */
double X[2]; /* Matrix X */
double det_C0_C1; /* Determinants of matrices */
double det_C0_X;
double det_X_C1;
double alpha_l; /* Alpha values, left and right */
double alpha_r;
Vector2 tmp; /* Utility variable */
BezierCurve bezCurve; /* RETURN bezier curve ctl pts */
bezCurve = (Vector2 *)malloc(4 * sizeof(Vector2));
nPts = last - first + 1;
/* Create the C and X matrices */
C[0][0] = 0.0;
C[0][1] = 0.0;
C[1][0] = 0.0;
C[1][1] = 0.0;
X[0] = 0.0;
X[1] = 0.0;
for (i = 0; i < nPts; i++) {
/* Compute the A's */
A[0] = tHat1;
A[1] = tHat2;
V2Scale(&A[0], B1(uPrime[i]));
V2Scale(&A[1], B2(uPrime[i]));
C[0][0] += V2Dot(&A[0], &A[0]);
C[0][1] += V2Dot(&A[0], &A[1]);
// C[1][0] += V2Dot(&A[0], &A[1]);
C[1][0] = C[0][1];
C[1][1] += V2Dot(&A[1], &A[1]);
tmp = V2SubII(d[first + i],
V2AddII(V2ScaleIII(d[first], B0(uPrime[i])),
V2AddII(V2ScaleIII(d[first], B1(uPrime[i])),
V2AddII(V2ScaleIII(d[last], B2(uPrime[i])),
V2ScaleIII(d[last], B3(uPrime[i]))))));
X[0] += V2Dot(&A[0], &tmp);
X[1] += V2Dot(&A[1], &tmp);
}
/* Compute the determinants of C and X */
det_C0_C1 = C[0][0] * C[1][1] - C[1][0] * C[0][1];
det_C0_X = C[0][0] * X[1] - C[0][1] * X[0];
det_X_C1 = X[0] * C[1][1] - X[1] * C[0][1];
/* Finally, derive alpha values */
if (det_C0_C1 == 0.0) {
det_C0_C1 = (C[0][0] * C[1][1]) * 10.0e-12;
}
alpha_l = det_X_C1 / det_C0_C1;
alpha_r = det_C0_X / det_C0_C1;
/* If alpha negative, use the Wu/Barsky heuristic (see text) (if alpha is 0, you get coincident
* control points that lead to divide by zero in any subsequent NewtonRaphsonRootFind() call).
*/
if (alpha_l < 1.0e-6 || alpha_r < 1.0e-6) {
double dist = V2DistanceBetween2Points(&d[last], &d[first]) / 3.0;
bezCurve[0] = d[first];
bezCurve[3] = d[last];
V2Add(&(bezCurve[0]), V2Scale(&(tHat1), dist), &(bezCurve[1]));
V2Add(&(bezCurve[3]), V2Scale(&(tHat2), dist), &(bezCurve[2]));
return bezCurve;
}
/* First and last control points of the Bezier curve are positioned exactly at the first and last
* data points Control points 1 and 2 are positioned an alpha distance out on the tangent
* vectors, left and right, respectively
*/
bezCurve[0] = d[first];
bezCurve[3] = d[last];
V2Add(&bezCurve[0], V2Scale(&tHat1, alpha_l), &bezCurve[1]);
V2Add(&bezCurve[3], V2Scale(&tHat2, alpha_r), &bezCurve[2]);
return bezCurve;
}
2008-04-30 15:41:54 +00:00
/*
* Reparameterize:
* Given set of points and their parameterization, try to find a better parameterization.
* Vector2 *d; Array of digitized points
* int first, last; Indices defining region
* double *u; Current parameter values
* BezierCurve bezCurve; Current fitted curve
2008-04-30 15:41:54 +00:00
*/
static double *Reparameterize(Vector2 *d, int first, int last, double *u, BezierCurve bezCurve)
{
int nPts = last - first + 1;
int i;
double *uPrime; /* New parameter values */
uPrime = (double *)malloc(nPts * sizeof(double));
for (i = first; i <= last; i++) {
uPrime[i - first] = NewtonRaphsonRootFind(bezCurve, d[i], u[i - first]);
}
return uPrime;
}
2008-04-30 15:41:54 +00:00
/*
* NewtonRaphsonRootFind:
* Use Newton-Raphson iteration to find better root.
* BezierCurve Q; Current fitted curve
* Vector2 P; Digitized point
* double u; Parameter value for "P"
2008-04-30 15:41:54 +00:00
*/
static double NewtonRaphsonRootFind(BezierCurve Q, Vector2 P, double u)
{
double numerator, denominator;
Vector2 Q1[3], Q2[2]; /* Q' and Q'' */
Vector2 Q_u, Q1_u, Q2_u; /* u evaluated at Q, Q', & Q'' */
double uPrime; /* Improved u */
int i;
/* Compute Q(u) */
Q_u = BezierII(3, Q, u);
/* Generate control vertices for Q' */
for (i = 0; i <= 2; i++) {
Q1[i][0] = (Q[i + 1][0] - Q[i][0]) * 3.0;
Q1[i][1] = (Q[i + 1][1] - Q[i][1]) * 3.0;
}
/* Generate control vertices for Q'' */
for (i = 0; i <= 1; i++) {
Q2[i][0] = (Q1[i + 1][0] - Q1[i][0]) * 2.0;
Q2[i][1] = (Q1[i + 1][1] - Q1[i][1]) * 2.0;
}
/* Compute Q'(u) and Q''(u) */
Q1_u = BezierII(2, Q1, u);
Q2_u = BezierII(1, Q2, u);
/* Compute f(u)/f'(u) */
numerator = (Q_u[0] - P[0]) * (Q1_u[0]) + (Q_u[1] - P[1]) * (Q1_u[1]);
denominator = (Q1_u[0]) * (Q1_u[0]) + (Q1_u[1]) * (Q1_u[1]) + (Q_u[0] - P[0]) * (Q2_u[0]) +
(Q_u[1] - P[1]) * (Q2_u[1]);
/* u = u - f(u)/f'(u) */
if (denominator == 0) { // FIXME
return u;
}
uPrime = u - (numerator / denominator);
return uPrime;
2008-04-30 15:41:54 +00:00
}
/*
* Bezier:
* Evaluate a Bezier curve at a particular parameter value
* int degree; The degree of the bezier curve
* Vector2 *V; Array of control points
* double t; Parametric value to find point for
2008-04-30 15:41:54 +00:00
*/
static Vector2 BezierII(int degree, Vector2 *V, double t)
{
int i, j;
Vector2 Q; /* Point on curve at parameter t */
Vector2 *Vtemp; /* Local copy of control points */
/* Copy array */
Vtemp = (Vector2 *)malloc((unsigned)((degree + 1) * sizeof(Vector2)));
for (i = 0; i <= degree; i++) {
Vtemp[i] = V[i];
}
/* Triangle computation */
for (i = 1; i <= degree; i++) {
for (j = 0; j <= degree - i; j++) {
Vtemp[j][0] = (1.0 - t) * Vtemp[j][0] + t * Vtemp[j + 1][0];
Vtemp[j][1] = (1.0 - t) * Vtemp[j][1] + t * Vtemp[j + 1][1];
}
}
Q = Vtemp[0];
free((void *)Vtemp);
return Q;
2008-04-30 15:41:54 +00:00
}
/*
* B0, B1, B2, B3:
* Bezier multipliers
2008-04-30 15:41:54 +00:00
*/
static double B0(double u)
{
double tmp = 1.0 - u;
return (tmp * tmp * tmp);
2008-04-30 15:41:54 +00:00
}
static double B1(double u)
{
double tmp = 1.0 - u;
return (3 * u * (tmp * tmp));
2008-04-30 15:41:54 +00:00
}
static double B2(double u)
{
double tmp = 1.0 - u;
return (3 * u * u * tmp);
2008-04-30 15:41:54 +00:00
}
static double B3(double u)
{
return (u * u * u);
2008-04-30 15:41:54 +00:00
}
/*
* ComputeLeftTangent, ComputeRightTangent, ComputeCenterTangent:
* Approximate unit tangents at endpoints and "center" of digitized curve
*/
/* Vector2 *d; Digitized points
* int end; Index to "left" end of region
2008-04-30 15:41:54 +00:00
*/
static Vector2 ComputeLeftTangent(Vector2 *d, int end)
{
Vector2 tHat1;
tHat1 = V2SubII(d[end + 1], d[end]);
tHat1 = *V2Normalize(&tHat1);
return tHat1;
2008-04-30 15:41:54 +00:00
}
/* Vector2 *d; Digitized points
* int end; Index to "right" end of region
*/
2008-04-30 15:41:54 +00:00
static Vector2 ComputeRightTangent(Vector2 *d, int end)
{
Vector2 tHat2;
tHat2 = V2SubII(d[end - 1], d[end]);
tHat2 = *V2Normalize(&tHat2);
return tHat2;
2008-04-30 15:41:54 +00:00
}
/* Vector2 *d; Digitized points
* int end; Index to point inside region
*/
2008-04-30 15:41:54 +00:00
static Vector2 ComputeCenterTangent(Vector2 *d, int center)
{
Vector2 V1, V2, tHatCenter;
V1 = V2SubII(d[center - 1], d[center]);
V2 = V2SubII(d[center], d[center + 1]);
tHatCenter[0] = (V1[0] + V2[0]) / 2.0;
tHatCenter[1] = (V1[1] + V2[1]) / 2.0;
tHatCenter = *V2Normalize(&tHatCenter);
/* avoid numerical singularity in the special case when V1 == -V2 */
if (V2Length(&tHatCenter) < M_EPSILON) {
tHatCenter = *V2Normalize(&V1);
}
return tHatCenter;
2008-04-30 15:41:54 +00:00
}
/*
* ChordLengthParameterize:
* Assign parameter values to digitized points using relative distances between points.
* Vector2 *d; Array of digitized points
* int first, last; Indices defining region
2008-04-30 15:41:54 +00:00
*/
static double *ChordLengthParameterize(Vector2 *d, int first, int last)
{
int i;
double *u; /* Parameterization */
2008-04-30 15:41:54 +00:00
u = (double *)malloc((unsigned)(last - first + 1) * sizeof(double));
2008-04-30 15:41:54 +00:00
u[0] = 0.0;
for (i = first + 1; i <= last; i++) {
u[i - first] = u[i - first - 1] + V2DistanceBetween2Points(&d[i], &d[i - 1]);
}
2008-04-30 15:41:54 +00:00
for (i = first + 1; i <= last; i++) {
u[i - first] = u[i - first] / u[last - first];
}
2008-04-30 15:41:54 +00:00
return u;
2008-04-30 15:41:54 +00:00
}
/*
* ComputeMaxError :
* Find the maximum squared distance of digitized points to fitted curve.
* Vector2 *d; Array of digitized points
* int first, last; Indices defining region
* BezierCurve bezCurve; Fitted Bezier curve
* double *u; Parameterization of points
* int *splitPoint; Point of maximum error
*/
static double ComputeMaxError(
Vector2 *d, int first, int last, BezierCurve bezCurve, double *u, int *splitPoint)
2008-04-30 15:41:54 +00:00
{
int i;
double maxDist; /* Maximum error */
double dist; /* Current error */
Vector2 P; /* Point on curve */
Vector2 v; /* Vector from point to curve */
*splitPoint = (last - first + 1) / 2;
maxDist = 0.0;
for (i = first + 1; i < last; i++) {
P = BezierII(3, bezCurve, u[i - first]);
v = V2SubII(P, d[i]);
dist = V2SquaredLength(&v);
if (dist >= maxDist) {
maxDist = dist;
*splitPoint = i;
}
}
return maxDist;
2008-04-30 15:41:54 +00:00
}
2008-04-30 15:41:54 +00:00
static Vector2 V2AddII(Vector2 a, Vector2 b)
{
Vector2 c;
c[0] = a[0] + b[0];
c[1] = a[1] + b[1];
return c;
2008-04-30 15:41:54 +00:00
}
2008-04-30 15:41:54 +00:00
static Vector2 V2ScaleIII(Vector2 v, double s)
{
Vector2 result;
result[0] = v[0] * s;
result[1] = v[1] * s;
return result;
2008-04-30 15:41:54 +00:00
}
static Vector2 V2SubII(Vector2 a, Vector2 b)
{
Vector2 c;
c[0] = a[0] - b[0];
c[1] = a[1] - b[1];
return c;
2008-04-30 15:41:54 +00:00
}
//------------------------- WRAPPER -----------------------------//
FitCurveWrapper::FitCurveWrapper()
{
}
FitCurveWrapper::~FitCurveWrapper()
{
_vertices.clear();
2008-04-30 15:41:54 +00:00
}
void FitCurveWrapper::DrawBezierCurve(int n, Vector2 *curve)
2008-04-30 15:41:54 +00:00
{
for (int i = 0; i <= n; ++i) {
_vertices.push_back(curve[i]);
}
2008-04-30 15:41:54 +00:00
}
void FitCurveWrapper::FitCurve(vector<Vec2d> &data, vector<Vec2d> &oCurve, double error)
2008-04-30 15:41:54 +00:00
{
int size = data.size();
Vector2 *d = new Vector2[size];
for (int i = 0; i < size; ++i) {
d[i][0] = data[i][0];
d[i][1] = data[i][1];
}
FitCurve(d, size, error);
delete[] d;
// copy results
for (vector<Vector2>::iterator v = _vertices.begin(), vend = _vertices.end(); v != vend; ++v) {
oCurve.emplace_back(v->x(), v->y());
}
2008-04-30 15:41:54 +00:00
}
void FitCurveWrapper::FitCurve(Vector2 *d, int nPts, double error)
{
Vector2 tHat1, tHat2; /* Unit tangent vectors at endpoints */
2008-04-30 15:41:54 +00:00
tHat1 = ComputeLeftTangent(d, 0);
tHat2 = ComputeRightTangent(d, nPts - 1);
FitCubic(d, 0, nPts - 1, tHat1, tHat2, error);
2008-04-30 15:41:54 +00:00
}
void FitCurveWrapper::FitCubic(
Vector2 *d, int first, int last, Vector2 tHat1, Vector2 tHat2, double error)
2008-04-30 15:41:54 +00:00
{
BezierCurve bezCurve; /* Control points of fitted Bezier curve */
double *u; /* Parameter values for point */
double *uPrime; /* Improved parameter values */
double maxError; /* Maximum fitting error */
int splitPoint; /* Point to split point set at */
int nPts; /* Number of points in subset */
double iterationError; /* Error below which you try iterating */
int maxIterations = 4; /* Max times to try iterating */
Vector2 tHatCenter; /* Unit tangent vector at splitPoint */
int i;
iterationError = error * error;
nPts = last - first + 1;
/* Use heuristic if region only has two points in it */
if (nPts == 2) {
double dist = V2DistanceBetween2Points(&d[last], &d[first]) / 3.0;
bezCurve = (Vector2 *)malloc(4 * sizeof(Vector2));
bezCurve[0] = d[first];
bezCurve[3] = d[last];
V2Add(&bezCurve[0], V2Scale(&tHat1, dist), &bezCurve[1]);
V2Add(&bezCurve[3], V2Scale(&tHat2, dist), &bezCurve[2]);
DrawBezierCurve(3, bezCurve);
free((void *)bezCurve);
return;
}
/* Parameterize points, and attempt to fit curve */
u = ChordLengthParameterize(d, first, last);
bezCurve = GenerateBezier(d, first, last, u, tHat1, tHat2);
/* Find max deviation of points to fitted curve */
maxError = ComputeMaxError(d, first, last, bezCurve, u, &splitPoint);
if (maxError < error) {
DrawBezierCurve(3, bezCurve);
free((void *)u);
free((void *)bezCurve);
return;
}
/* If error not too large, try some reparameterization and iteration */
if (maxError < iterationError) {
for (i = 0; i < maxIterations; i++) {
uPrime = Reparameterize(d, first, last, u, bezCurve);
free((void *)u);
free((void *)bezCurve);
u = uPrime;
bezCurve = GenerateBezier(d, first, last, u, tHat1, tHat2);
maxError = ComputeMaxError(d, first, last, bezCurve, u, &splitPoint);
if (maxError < error) {
DrawBezierCurve(3, bezCurve);
free((void *)u);
free((void *)bezCurve);
return;
}
}
}
/* Fitting failed -- split at max error point and fit recursively */
free((void *)u);
free((void *)bezCurve);
tHatCenter = ComputeCenterTangent(d, splitPoint);
FitCubic(d, first, splitPoint, tHat1, tHatCenter, error);
V2Negate(&tHatCenter);
FitCubic(d, splitPoint, last, tHatCenter, tHat2, error);
2008-04-30 15:41:54 +00:00
}
Attempt to fix a potential name conflict between Freestyle and the compositor. A crash in the Freestyle renderer was reported by Ton on IRC with a stack trace below. Note that #2 is in Freestyle, whereas #1 is in the compositor. The problem was observed in a debug build on OS X 10.7 (gcc 4.2, openmp disabled, no llvm). ---------------------------------------------------------------------- Program received signal EXC_BAD_ACCESS, Could not access memory. Reason: 13 at address: 0x0000000000000000 [Switching to process 72386 thread 0xf303] 0x0000000100c129f3 in NodeBase::~NodeBase (this=0x10e501c80) at COM_NodeBase.cpp:43 43 delete (this->m_outputsockets.back()); Current language: auto; currently c++ (gdb) where #0 0x0000000100c129f3 in NodeBase::~NodeBase (this=0x10e501c80) at COM_NodeBase.cpp:43 #1 0x0000000100c29066 in Node::~Node (this=0x10e501c80) at COM_Node.h:49 #2 0x000000010089c273 in NodeShape::~NodeShape (this=0x10e501c80) at NodeShape.cpp:43 #3 0x000000010089910b in NodeGroup::destroy (this=0x10e501da0) at NodeGroup.cpp:61 #4 0x00000001008990cd in NodeGroup::destroy (this=0x10e5014b0) at NodeGroup.cpp:59 #5 0x00000001008990cd in NodeGroup::destroy (this=0x114e18da0) at NodeGroup.cpp:59 #6 0x00000001007e6602 in Controller::ClearRootNode (this=0x114e19640) at Controller.cpp:329 #7 0x00000001007ea52e in Controller::LoadMesh (this=0x114e19640, re=0x10aba4638, srl=0x1140f5258) at Controller.cpp:302 #8 0x00000001008030ad in prepare (re=0x10aba4638, srl=0x1140f5258) at FRS_freestyle.cpp:302 #9 0x000000010080457a in FRS_do_stroke_rendering (re=0x10aba4638, srl=0x1140f5258) at FRS_freestyle.cpp:600 #10 0x00000001006aeb9d in add_freestyle (re=0x10aba4638) at pipeline.c:1584 #11 0x00000001006aceb7 in do_render_3d (re=0x10aba4638) at pipeline.c:1094 #12 0x00000001006ae061 in do_render_fields_blur_3d (re=0x10aba4638) at pipeline.c:1367 #13 0x00000001006afa16 in do_render_composite_fields_blur_3d (re=0x10aba4638) at pipeline.c:1815 #14 0x00000001006b04e4 in do_render_all_options (re=0x10aba4638) at pipeline.c:2021 ---------------------------------------------------------------------- Apparently a name conflict between the two Blender modules is taking place. The present commit hence intends to address it by putting all the Freestyle C++ classes in the namespace 'Freestyle'. This revision will also prevent potential name conflicts with other Blender modules in the future. Special thanks to Lukas Toenne for the help with C++ namespace.
2013-04-09 00:46:49 +00:00
} /* namespace Freestyle */