2014-06-04 16:25:24 +02:00
|
|
|
|
|
|
|
// Only in 3d
|
|
|
|
|
|
|
|
template<typename T, typename ProdType>
|
|
|
|
static T project_tool(T *vertex_value, T *u, T *u0)
|
|
|
|
{
|
|
|
|
T ret0 = 0;
|
2014-06-13 18:28:05 +02:00
|
|
|
for (unsigned int i = 0; i < 8; i++)
|
2014-06-04 16:25:24 +02:00
|
|
|
{
|
2014-06-13 18:28:05 +02:00
|
|
|
unsigned int c[3] = { i & 1, (i>>1)&1, (i>>2)&1 };
|
2014-06-04 16:25:24 +02:00
|
|
|
int epsilon[3];
|
|
|
|
T ret = 0;
|
|
|
|
|
|
|
|
for (int q = 0; q < 3; q++)
|
2014-06-13 18:28:05 +02:00
|
|
|
epsilon[q] = 2*c[q]-1;
|
2014-06-04 16:25:24 +02:00
|
|
|
|
|
|
|
for (int q = 0; q < ProdType::numProducts; q++)
|
|
|
|
ret += ProdType::product(u, u0, epsilon, q);
|
|
|
|
ret *= vertex_value[i];
|
|
|
|
ret0 += ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-05 14:07:57 +02:00
|
|
|
template<typename T>
|
2014-06-05 17:36:57 +02:00
|
|
|
static inline T get_u0(const T& u0, int epsilon)
|
2014-06-05 14:07:57 +02:00
|
|
|
{
|
2014-06-13 18:28:05 +02:00
|
|
|
return (1-epsilon)/2 + epsilon*u0;
|
|
|
|
// return (epsilon > 0) ? u0 : (1-u0);
|
2014-06-05 14:07:57 +02:00
|
|
|
}
|
|
|
|
|
2014-06-04 16:25:24 +02:00
|
|
|
template<typename T>
|
|
|
|
struct ProductTerm0
|
|
|
|
{
|
|
|
|
static const int numProducts = 1;
|
|
|
|
|
2014-06-05 17:36:57 +02:00
|
|
|
static inline T product(T *u, T *u0, int *epsilon, int q)
|
2014-06-04 16:25:24 +02:00
|
|
|
{
|
|
|
|
T a = 1;
|
|
|
|
|
2014-06-13 18:28:05 +02:00
|
|
|
for (unsigned int r = 0; r < 3; r++)
|
2014-06-05 14:07:57 +02:00
|
|
|
a *= get_u0(u0[r], epsilon[r]);
|
2014-06-04 16:25:24 +02:00
|
|
|
return a;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-06-05 14:07:57 +02:00
|
|
|
|
2014-06-04 16:25:24 +02:00
|
|
|
template<typename T>
|
|
|
|
struct ProductTerm1
|
|
|
|
{
|
|
|
|
static const int numProducts = 3;
|
|
|
|
|
|
|
|
static T product(T *u, T *u0, int *epsilon, int q)
|
|
|
|
{
|
|
|
|
T a = 1;
|
2014-06-13 18:28:05 +02:00
|
|
|
T G[3];
|
2014-06-04 16:25:24 +02:00
|
|
|
|
2014-06-13 18:28:05 +02:00
|
|
|
for (unsigned int r = 0; r < 3; r++)
|
2014-06-04 16:25:24 +02:00
|
|
|
{
|
2014-06-05 14:07:57 +02:00
|
|
|
G[r] = get_u0(u0[r], epsilon[r]);
|
2014-06-04 16:25:24 +02:00
|
|
|
}
|
|
|
|
|
2014-06-13 18:28:05 +02:00
|
|
|
T F[3] = { G[1]*G[2], G[0]*G[2], G[0]*G[1] };
|
2014-06-04 16:25:24 +02:00
|
|
|
|
2014-06-05 14:07:57 +02:00
|
|
|
return F[q] * u[q] * epsilon[q];
|
2014-06-04 16:25:24 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
struct ProductTerm2
|
|
|
|
{
|
|
|
|
static const int numProducts = 3;
|
|
|
|
|
2014-06-05 17:36:57 +02:00
|
|
|
static inline T product(T *u, T *u0, int *epsilon, int q)
|
2014-06-04 16:25:24 +02:00
|
|
|
{
|
|
|
|
T a = 1;
|
2014-06-13 18:28:05 +02:00
|
|
|
T G[3];
|
2014-06-04 16:25:24 +02:00
|
|
|
|
2014-06-13 18:28:05 +02:00
|
|
|
for (unsigned int r = 0; r < 3; r++)
|
2014-06-04 16:25:24 +02:00
|
|
|
{
|
2014-06-05 14:07:57 +02:00
|
|
|
G[r] = get_u0(u0[r], epsilon[r]);
|
2014-06-04 16:25:24 +02:00
|
|
|
}
|
|
|
|
|
2014-06-13 18:28:05 +02:00
|
|
|
T F[3] = { epsilon[1]*epsilon[2]*u[1]*u[2], epsilon[0]*epsilon[2]*u[0]*u[2], epsilon[0]*epsilon[1]*u[0]*u[1] };
|
2014-06-04 16:25:24 +02:00
|
|
|
|
2014-06-05 14:07:57 +02:00
|
|
|
return F[q] * G[q];
|
2014-06-04 16:25:24 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-06-05 14:07:57 +02:00
|
|
|
|
2014-06-04 16:25:24 +02:00
|
|
|
template<typename T>
|
|
|
|
struct ProductTerm3
|
|
|
|
{
|
|
|
|
static const int numProducts = 1;
|
|
|
|
|
2014-06-05 17:36:57 +02:00
|
|
|
static inline T product(T *u, T *u0, int *epsilon, int q)
|
2014-06-04 16:25:24 +02:00
|
|
|
{
|
2014-06-05 14:07:57 +02:00
|
|
|
return epsilon[0]*epsilon[1]*epsilon[2]*u[0]*u[1]*u[2];
|
2014-06-04 16:25:24 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
T compute_projection(T *vertex_value, T *u, T *u0, T rho)
|
|
|
|
{
|
|
|
|
T ret;
|
|
|
|
|
|
|
|
ret = project_tool<T, ProductTerm0<T> >(vertex_value, u, u0) * rho;
|
2014-06-05 17:36:57 +02:00
|
|
|
ret += project_tool<T, ProductTerm1<T> >(vertex_value, u, u0) * rho * rho / 2;
|
|
|
|
ret += project_tool<T, ProductTerm2<T> >(vertex_value, u, u0) * rho * rho * rho / 3;
|
|
|
|
ret += project_tool<T, ProductTerm3<T> >(vertex_value, u, u0) * rho * rho * rho * rho / 4;
|
2014-06-04 16:25:24 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
template
|
|
|
|
double compute_projection(double *vertex_value, double *u, double *u0, double rho);
|
|
|
|
|