2011-02-23 10:52:22 +00:00
|
|
|
/*
|
2011-02-09 01:27:46 +00:00
|
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2011 Blender Foundation.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
|
|
*/
|
|
|
|
|
2011-02-27 20:40:57 +00:00
|
|
|
/** \file blender/blenkernel/intern/mesh_validate.c
|
|
|
|
* \ingroup bke
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2011-02-09 01:27:46 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
#include "DNA_mesh_types.h"
|
|
|
|
#include "DNA_meshdata_types.h"
|
2012-03-15 20:10:07 +00:00
|
|
|
#include "DNA_object_types.h"
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2013-05-28 19:35:26 +00:00
|
|
|
#include "BLI_sys_types.h"
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2013-04-22 12:00:37 +00:00
|
|
|
#include "BLI_utildefines.h"
|
2011-02-09 01:27:46 +00:00
|
|
|
#include "BLI_edgehash.h"
|
2011-12-01 19:21:58 +00:00
|
|
|
#include "BLI_math_base.h"
|
2013-04-27 12:51:23 +00:00
|
|
|
#include "BLI_math_vector.h"
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
#include "BKE_deform.h"
|
2011-02-09 02:28:11 +00:00
|
|
|
#include "BKE_DerivedMesh.h"
|
2012-03-15 20:10:07 +00:00
|
|
|
#include "BKE_mesh.h"
|
2011-02-09 02:28:11 +00:00
|
|
|
|
2017-04-06 16:11:50 +02:00
|
|
|
#include "DEG_depsgraph.h"
|
|
|
|
|
2011-02-09 01:27:46 +00:00
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
2013-09-09 02:11:44 +00:00
|
|
|
/* loop v/e are unsigned, so using max uint_32 value as invalid marker... */
|
|
|
|
#define INVALID_LOOP_EDGE_MARKER 4294967295u
|
|
|
|
|
|
|
|
|
|
|
|
/** \name Internal functions
|
|
|
|
* \{ */
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2012-06-28 09:08:11 +00:00
|
|
|
typedef union {
|
|
|
|
uint32_t verts[2];
|
|
|
|
int64_t edval;
|
|
|
|
} EdgeUUID;
|
|
|
|
|
|
|
|
typedef struct SortFace {
|
|
|
|
EdgeUUID es[4];
|
|
|
|
unsigned int index;
|
|
|
|
} SortFace;
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Used to detect polys (faces) using exactly the same vertices. */
|
|
|
|
/* Used to detect loops used by no (disjoint) or more than one (intersect) polys. */
|
|
|
|
typedef struct SortPoly {
|
|
|
|
int *verts;
|
|
|
|
int numverts;
|
|
|
|
int loopstart;
|
|
|
|
unsigned int index;
|
2014-04-01 11:34:00 +11:00
|
|
|
bool invalid; /* Poly index. */
|
2012-03-15 20:10:07 +00:00
|
|
|
} SortPoly;
|
|
|
|
|
2012-06-28 09:08:11 +00:00
|
|
|
static void edge_store_assign(uint32_t verts[2], const uint32_t v1, const uint32_t v2)
|
|
|
|
{
|
|
|
|
if (v1 < v2) {
|
|
|
|
verts[0] = v1;
|
|
|
|
verts[1] = v2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
verts[0] = v2;
|
|
|
|
verts[1] = v1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void edge_store_from_mface_quad(EdgeUUID es[4], MFace *mf)
|
|
|
|
{
|
|
|
|
edge_store_assign(es[0].verts, mf->v1, mf->v2);
|
|
|
|
edge_store_assign(es[1].verts, mf->v2, mf->v3);
|
|
|
|
edge_store_assign(es[2].verts, mf->v3, mf->v4);
|
|
|
|
edge_store_assign(es[3].verts, mf->v4, mf->v1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void edge_store_from_mface_tri(EdgeUUID es[4], MFace *mf)
|
|
|
|
{
|
|
|
|
edge_store_assign(es[0].verts, mf->v1, mf->v2);
|
|
|
|
edge_store_assign(es[1].verts, mf->v2, mf->v3);
|
|
|
|
edge_store_assign(es[2].verts, mf->v3, mf->v1);
|
|
|
|
es[3].verts[0] = es[3].verts[1] = UINT_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int int64_cmp(const void *v1, const void *v2)
|
|
|
|
{
|
|
|
|
const int64_t x1 = *(const int64_t *)v1;
|
|
|
|
const int64_t x2 = *(const int64_t *)v2;
|
|
|
|
|
|
|
|
if (x1 > x2) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (x1 < x2) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int search_face_cmp(const void *v1, const void *v2)
|
|
|
|
{
|
|
|
|
const SortFace *sfa = v1, *sfb = v2;
|
|
|
|
|
|
|
|
if (sfa->es[0].edval > sfb->es[0].edval) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (sfa->es[0].edval < sfb->es[0].edval) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (sfa->es[1].edval > sfb->es[1].edval) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (sfa->es[1].edval < sfb->es[1].edval) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (sfa->es[2].edval > sfb->es[2].edval) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (sfa->es[2].edval < sfb->es[2].edval) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (sfa->es[3].edval > sfb->es[3].edval) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (sfa->es[3].edval < sfb->es[3].edval) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* TODO check there is not some standard define of this somewhere! */
|
|
|
|
static int int_cmp(const void *v1, const void *v2)
|
2011-02-09 15:13:20 +00:00
|
|
|
{
|
2012-04-18 09:16:30 +00:00
|
|
|
return *(int *)v1 > *(int *)v2 ? 1 : *(int *)v1 < *(int *)v2 ? -1 : 0;
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
static int search_poly_cmp(const void *v1, const void *v2)
|
2011-02-09 01:27:46 +00:00
|
|
|
{
|
2012-03-15 20:10:07 +00:00
|
|
|
const SortPoly *sp1 = v1, *sp2 = v2;
|
|
|
|
const int max_idx = sp1->numverts > sp2->numverts ? sp2->numverts : sp1->numverts;
|
2014-07-25 16:45:26 +02:00
|
|
|
int idx;
|
2012-03-15 20:10:07 +00:00
|
|
|
|
|
|
|
/* Reject all invalid polys at end of list! */
|
|
|
|
if (sp1->invalid || sp2->invalid)
|
2014-07-25 16:45:26 +02:00
|
|
|
return sp1->invalid ? (sp2->invalid ? 0 : 1) : -1;
|
|
|
|
/* Else, sort on first non-equal verts (remember verts of valid polys are sorted). */
|
|
|
|
for (idx = 0; idx < max_idx; idx++) {
|
2015-11-23 11:27:02 +11:00
|
|
|
const int v1_i = sp1->verts[idx];
|
|
|
|
const int v2_i = sp2->verts[idx];
|
|
|
|
if (v1_i != v2_i) {
|
|
|
|
return (v1_i > v2_i) ? 1 : -1;
|
2014-07-25 16:45:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return sp1->numverts > sp2->numverts ? 1 : sp1->numverts < sp2->numverts ? -1 : 0;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
static int search_polyloop_cmp(const void *v1, const void *v2)
|
2011-02-09 01:27:46 +00:00
|
|
|
{
|
2012-03-15 20:10:07 +00:00
|
|
|
const SortPoly *sp1 = v1, *sp2 = v2;
|
2011-02-10 12:34:52 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Reject all invalid polys at end of list! */
|
|
|
|
if (sp1->invalid || sp2->invalid)
|
|
|
|
return sp1->invalid && sp2->invalid ? 0 : sp1->invalid ? 1 : -1;
|
|
|
|
/* Else, sort on loopstart. */
|
|
|
|
return sp1->loopstart > sp2->loopstart ? 1 : sp1->loopstart < sp2->loopstart ? -1 : 0;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
2013-09-09 02:11:44 +00:00
|
|
|
/** \} */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/** \name Mesh Validation
|
|
|
|
* \{ */
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
#define PRINT_MSG(...) (void) \
|
|
|
|
( \
|
|
|
|
((do_verbose) ? printf(__VA_ARGS__) : 0))
|
|
|
|
|
|
|
|
#define PRINT_ERR(...) (void) \
|
|
|
|
(is_valid = false, \
|
|
|
|
((do_verbose) ? printf(__VA_ARGS__) : 0))
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate the mesh, \a do_fixes requires \a mesh to be non-null.
|
|
|
|
*
|
|
|
|
* \return false if no changes needed to be made.
|
|
|
|
*/
|
|
|
|
bool BKE_mesh_validate_arrays(Mesh *mesh,
|
|
|
|
MVert *mverts, unsigned int totvert,
|
|
|
|
MEdge *medges, unsigned int totedge,
|
|
|
|
MFace *mfaces, unsigned int totface,
|
|
|
|
MLoop *mloops, unsigned int totloop,
|
|
|
|
MPoly *mpolys, unsigned int totpoly,
|
|
|
|
MDeformVert *dverts, /* assume totvert length */
|
|
|
|
const bool do_verbose, const bool do_fixes,
|
2013-11-26 06:39:14 +11:00
|
|
|
bool *r_changed)
|
2011-02-09 01:27:46 +00:00
|
|
|
{
|
2015-02-24 13:08:07 +11:00
|
|
|
# define REMOVE_EDGE_TAG(_me) { _me->v2 = _me->v1; free_flag.edges = do_fixes; } (void)0
|
2012-04-18 09:16:30 +00:00
|
|
|
# define IS_REMOVED_EDGE(_me) (_me->v2 == _me->v1)
|
2012-03-15 20:10:07 +00:00
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
# define REMOVE_LOOP_TAG(_ml) { _ml->e = INVALID_LOOP_EDGE_MARKER; free_flag.polyloops = do_fixes; } (void)0
|
|
|
|
# define REMOVE_POLY_TAG(_mp) { _mp->totloop *= -1; free_flag.polyloops = do_fixes; } (void)0
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
MVert *mv = mverts;
|
|
|
|
MEdge *me;
|
|
|
|
MLoop *ml;
|
|
|
|
MPoly *mp;
|
|
|
|
unsigned int i, j;
|
|
|
|
int *v;
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
bool is_valid = true;
|
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
union {
|
|
|
|
struct {
|
|
|
|
int verts : 1;
|
|
|
|
int verts_weight : 1;
|
2015-10-07 14:38:36 +11:00
|
|
|
int loops_edge : 1;
|
2015-02-24 13:08:07 +11:00
|
|
|
};
|
|
|
|
int as_flag;
|
|
|
|
} fix_flag;
|
|
|
|
|
|
|
|
union {
|
|
|
|
struct {
|
|
|
|
int edges : 1;
|
|
|
|
int faces : 1;
|
|
|
|
/* This regroups loops and polys! */
|
|
|
|
int polyloops : 1;
|
|
|
|
int mselect : 1;
|
|
|
|
};
|
|
|
|
int as_flag;
|
|
|
|
} free_flag;
|
|
|
|
|
|
|
|
union {
|
|
|
|
struct {
|
|
|
|
int edges : 1;
|
|
|
|
};
|
|
|
|
int as_flag;
|
|
|
|
} recalc_flag;
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2013-08-24 14:40:15 +00:00
|
|
|
EdgeHash *edge_hash = BLI_edgehash_new_ex(__func__, totedge);
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
BLI_assert(!(do_fixes && mesh == NULL));
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
fix_flag.as_flag = 0;
|
|
|
|
free_flag.as_flag = 0;
|
|
|
|
recalc_flag.as_flag = 0;
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_MSG("%s: verts(%u), edges(%u), loops(%u), polygons(%u)\n",
|
|
|
|
__func__, totvert, totedge, totloop, totpoly);
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (totedge == 0 && totpoly != 0) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tLogical error, %u polygons and 0 edges\n", totpoly);
|
2015-02-24 13:08:07 +11:00
|
|
|
recalc_flag.edges = do_fixes;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2014-08-19 18:49:54 +10:00
|
|
|
for (i = 0; i < totvert; i++, mv++) {
|
2014-04-01 11:34:00 +11:00
|
|
|
bool fix_normal = true;
|
2011-12-01 19:21:58 +00:00
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
for (j = 0; j < 3; j++) {
|
2016-05-16 00:48:02 +02:00
|
|
|
if (!isfinite(mv->co[j])) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tVertex %u: has invalid coordinate\n", i);
|
2011-12-01 19:21:58 +00:00
|
|
|
|
2011-12-09 07:23:17 +00:00
|
|
|
if (do_fixes) {
|
2012-03-15 20:10:07 +00:00
|
|
|
zero_v3(mv->co);
|
2011-12-09 07:23:17 +00:00
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
fix_flag.verts = true;
|
2011-12-09 07:23:17 +00:00
|
|
|
}
|
2011-12-01 19:21:58 +00:00
|
|
|
}
|
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
if (mv->no[j] != 0)
|
2014-04-01 11:34:00 +11:00
|
|
|
fix_normal = false;
|
2011-12-01 19:21:58 +00:00
|
|
|
}
|
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (fix_normal) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tVertex %u: has zero normal, assuming Z-up normal\n", i);
|
2011-12-09 07:23:17 +00:00
|
|
|
if (do_fixes) {
|
2012-04-18 09:16:30 +00:00
|
|
|
mv->no[2] = SHRT_MAX;
|
2015-02-24 13:08:07 +11:00
|
|
|
fix_flag.verts = true;
|
2011-12-09 07:23:17 +00:00
|
|
|
}
|
2011-12-01 19:21:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
for (i = 0, me = medges; i < totedge; i++, me++) {
|
2014-04-11 11:25:41 +10:00
|
|
|
bool remove = false;
|
2015-02-26 17:47:41 +11:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (me->v1 == me->v2) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tEdge %u: has matching verts, both %u\n", i, me->v1);
|
2012-04-18 09:16:30 +00:00
|
|
|
remove = do_fixes;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
if (me->v1 >= totvert) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tEdge %u: v1 index out of range, %u\n", i, me->v1);
|
2012-04-18 09:16:30 +00:00
|
|
|
remove = do_fixes;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
if (me->v2 >= totvert) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tEdge %u: v2 index out of range, %u\n", i, me->v2);
|
2012-04-18 09:16:30 +00:00
|
|
|
remove = do_fixes;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2015-02-26 17:47:41 +11:00
|
|
|
if ((me->v1 != me->v2) && BLI_edgehash_haskey(edge_hash, me->v1, me->v2)) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tEdge %u: is a duplicate of %d\n", i,
|
|
|
|
GET_INT_FROM_POINTER(BLI_edgehash_lookup(edge_hash, me->v1, me->v2)));
|
2012-04-18 09:16:30 +00:00
|
|
|
remove = do_fixes;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2014-04-01 11:34:00 +11:00
|
|
|
if (remove == false) {
|
2015-02-26 17:47:41 +11:00
|
|
|
if (me->v1 != me->v2) {
|
|
|
|
BLI_edgehash_insert(edge_hash, me->v1, me->v2, SET_INT_IN_POINTER(i));
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-03-15 20:10:07 +00:00
|
|
|
REMOVE_EDGE_TAG(me);
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 09:08:11 +00:00
|
|
|
if (mfaces && !mpolys) {
|
2015-02-24 13:08:07 +11:00
|
|
|
# define REMOVE_FACE_TAG(_mf) { _mf->v3 = 0; free_flag.faces = do_fixes; } (void)0
|
2012-06-28 09:08:11 +00:00
|
|
|
# define CHECK_FACE_VERT_INDEX(a, b) \
|
|
|
|
if (mf->a == mf->b) { \
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR(" face %u: verts invalid, " STRINGIFY(a) "/" STRINGIFY(b) " both %u\n", i, mf->a); \
|
2012-06-28 09:08:11 +00:00
|
|
|
remove = do_fixes; \
|
|
|
|
} (void)0
|
|
|
|
# define CHECK_FACE_EDGE(a, b) \
|
|
|
|
if (!BLI_edgehash_haskey(edge_hash, mf->a, mf->b)) { \
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR(" face %u: edge " STRINGIFY(a) "/" STRINGIFY(b) \
|
|
|
|
" (%u,%u) is missing edge data\n", i, mf->a, mf->b); \
|
2015-02-24 13:08:07 +11:00
|
|
|
recalc_flag.edges = do_fixes; \
|
2012-07-05 13:02:42 +00:00
|
|
|
} (void)0
|
2012-06-28 09:08:11 +00:00
|
|
|
|
|
|
|
MFace *mf;
|
|
|
|
MFace *mf_prev;
|
|
|
|
|
|
|
|
SortFace *sort_faces = MEM_callocN(sizeof(SortFace) * totface, "search faces");
|
|
|
|
SortFace *sf;
|
|
|
|
SortFace *sf_prev;
|
|
|
|
unsigned int totsortface = 0;
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("No Polys, only tesselated Faces\n");
|
2012-07-05 13:02:42 +00:00
|
|
|
|
2012-06-28 09:08:11 +00:00
|
|
|
for (i = 0, mf = mfaces, sf = sort_faces; i < totface; i++, mf++) {
|
2014-04-11 11:25:41 +10:00
|
|
|
bool remove = false;
|
2012-06-28 09:08:11 +00:00
|
|
|
int fidx;
|
|
|
|
unsigned int fv[4];
|
|
|
|
|
|
|
|
fidx = mf->v4 ? 3 : 2;
|
|
|
|
do {
|
|
|
|
fv[fidx] = *(&(mf->v1) + fidx);
|
|
|
|
if (fv[fidx] >= totvert) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tFace %u: 'v%d' index out of range, %u\n", i, fidx + 1, fv[fidx]);
|
2012-06-28 09:08:11 +00:00
|
|
|
remove = do_fixes;
|
|
|
|
}
|
|
|
|
} while (fidx--);
|
|
|
|
|
2014-04-01 11:34:00 +11:00
|
|
|
if (remove == false) {
|
2012-06-28 09:08:11 +00:00
|
|
|
if (mf->v4) {
|
|
|
|
CHECK_FACE_VERT_INDEX(v1, v2);
|
|
|
|
CHECK_FACE_VERT_INDEX(v1, v3);
|
|
|
|
CHECK_FACE_VERT_INDEX(v1, v4);
|
|
|
|
|
|
|
|
CHECK_FACE_VERT_INDEX(v2, v3);
|
|
|
|
CHECK_FACE_VERT_INDEX(v2, v4);
|
|
|
|
|
|
|
|
CHECK_FACE_VERT_INDEX(v3, v4);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
CHECK_FACE_VERT_INDEX(v1, v2);
|
|
|
|
CHECK_FACE_VERT_INDEX(v1, v3);
|
|
|
|
|
|
|
|
CHECK_FACE_VERT_INDEX(v2, v3);
|
|
|
|
}
|
|
|
|
|
2014-04-01 11:34:00 +11:00
|
|
|
if (remove == false) {
|
2012-06-28 09:08:11 +00:00
|
|
|
if (totedge) {
|
|
|
|
if (mf->v4) {
|
|
|
|
CHECK_FACE_EDGE(v1, v2);
|
|
|
|
CHECK_FACE_EDGE(v2, v3);
|
|
|
|
CHECK_FACE_EDGE(v3, v4);
|
|
|
|
CHECK_FACE_EDGE(v4, v1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
CHECK_FACE_EDGE(v1, v2);
|
|
|
|
CHECK_FACE_EDGE(v2, v3);
|
|
|
|
CHECK_FACE_EDGE(v3, v1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sf->index = i;
|
|
|
|
|
|
|
|
if (mf->v4) {
|
|
|
|
edge_store_from_mface_quad(sf->es, mf);
|
|
|
|
|
|
|
|
qsort(sf->es, 4, sizeof(int64_t), int64_cmp);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
edge_store_from_mface_tri(sf->es, mf);
|
|
|
|
qsort(sf->es, 3, sizeof(int64_t), int64_cmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
totsortface++;
|
|
|
|
sf++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (remove) {
|
|
|
|
REMOVE_FACE_TAG(mf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qsort(sort_faces, totsortface, sizeof(SortFace), search_face_cmp);
|
|
|
|
|
|
|
|
sf = sort_faces;
|
|
|
|
sf_prev = sf;
|
|
|
|
sf++;
|
|
|
|
|
|
|
|
for (i = 1; i < totsortface; i++, sf++) {
|
2014-04-11 11:25:41 +10:00
|
|
|
bool remove = false;
|
2012-06-28 09:08:11 +00:00
|
|
|
|
|
|
|
/* on a valid mesh, code below will never run */
|
|
|
|
if (memcmp(sf->es, sf_prev->es, sizeof(sf_prev->es)) == 0) {
|
|
|
|
mf = mfaces + sf->index;
|
|
|
|
|
|
|
|
if (do_verbose) {
|
|
|
|
mf_prev = mfaces + sf_prev->index;
|
|
|
|
|
|
|
|
if (mf->v4) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tFace %u & %u: are duplicates (%u,%u,%u,%u) (%u,%u,%u,%u)\n",
|
|
|
|
sf->index, sf_prev->index, mf->v1, mf->v2, mf->v3, mf->v4,
|
|
|
|
mf_prev->v1, mf_prev->v2, mf_prev->v3, mf_prev->v4);
|
2012-06-28 09:08:11 +00:00
|
|
|
}
|
|
|
|
else {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tFace %u & %u: are duplicates (%u,%u,%u) (%u,%u,%u)\n",
|
|
|
|
sf->index, sf_prev->index, mf->v1, mf->v2, mf->v3,
|
|
|
|
mf_prev->v1, mf_prev->v2, mf_prev->v3);
|
2012-06-28 09:08:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
remove = do_fixes;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
sf_prev = sf;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (remove) {
|
|
|
|
REMOVE_FACE_TAG(mf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(sort_faces);
|
|
|
|
|
|
|
|
# undef REMOVE_FACE_TAG
|
|
|
|
# undef CHECK_FACE_VERT_INDEX
|
|
|
|
# undef CHECK_FACE_EDGE
|
|
|
|
}
|
|
|
|
|
2014-01-17 17:35:03 +11:00
|
|
|
/* Checking loops and polys is a bit tricky, as they are quite intricate...
|
2012-03-15 20:10:07 +00:00
|
|
|
*
|
|
|
|
* Polys must have:
|
|
|
|
* - a valid loopstart value.
|
|
|
|
* - a valid totloop value (>= 3 and loopstart+totloop < me.totloop).
|
|
|
|
*
|
|
|
|
* Loops must have:
|
|
|
|
* - a valid v value.
|
|
|
|
* - a valid e value (corresponding to the edge it defines with the next loop in poly).
|
|
|
|
*
|
|
|
|
* Also, loops not used by polys can be discarded.
|
|
|
|
* And "intersecting" loops (i.e. loops used by more than one poly) are invalid,
|
2012-07-05 13:02:42 +00:00
|
|
|
* so be sure to leave at most one poly per loop!
|
2012-03-15 20:10:07 +00:00
|
|
|
*/
|
|
|
|
{
|
|
|
|
SortPoly *sort_polys = MEM_callocN(sizeof(SortPoly) * totpoly, "mesh validate's sort_polys");
|
|
|
|
SortPoly *prev_sp, *sp = sort_polys;
|
|
|
|
int prev_end;
|
2015-11-15 17:11:19 +01:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
for (i = 0, mp = mpolys; i < totpoly; i++, mp++, sp++) {
|
|
|
|
sp->index = i;
|
|
|
|
|
|
|
|
if (mp->loopstart < 0 || mp->totloop < 3) {
|
|
|
|
/* Invalid loop data. */
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tPoly %u is invalid (loopstart: %d, totloop: %d)\n",
|
2013-09-04 01:29:34 +00:00
|
|
|
sp->index, mp->loopstart, mp->totloop);
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
else if (mp->loopstart + mp->totloop > totloop) {
|
|
|
|
/* Invalid loop data. */
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tPoly %u uses loops out of range (loopstart: %d, loopend: %d, max nbr of loops: %u)\n",
|
2013-09-04 01:29:34 +00:00
|
|
|
sp->index, mp->loopstart, mp->loopstart + mp->totloop - 1, totloop - 1);
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Poly itself is valid, for now. */
|
|
|
|
int v1, v2; /* v1 is prev loop vert idx, v2 is current loop one. */
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = false;
|
2012-03-15 20:10:07 +00:00
|
|
|
sp->verts = v = MEM_mallocN(sizeof(int) * mp->totloop, "Vert idx of SortPoly");
|
|
|
|
sp->numverts = mp->totloop;
|
|
|
|
sp->loopstart = mp->loopstart;
|
|
|
|
|
2015-11-15 17:11:19 +01:00
|
|
|
/* Ideally we would only have to do that once on all vertices before we start checking each poly, but
|
|
|
|
* several polys can use same vert, so we have to ensure here all verts of current poly are cleared. */
|
|
|
|
for (j = 0, ml = &mloops[sp->loopstart]; j < mp->totloop; j++, ml++) {
|
|
|
|
if (ml->v < totvert) {
|
|
|
|
mverts[ml->v].flag &= ~ME_VERT_TMP_TAG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Test all poly's loops' vert idx. */
|
|
|
|
for (j = 0, ml = &mloops[sp->loopstart]; j < mp->totloop; j++, ml++, v++) {
|
|
|
|
if (ml->v >= totvert) {
|
|
|
|
/* Invalid vert idx. */
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tLoop %u has invalid vert reference (%u)\n", sp->loopstart + j, ml->v);
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
2015-11-15 17:11:19 +01:00
|
|
|
else if (mverts[ml->v].flag & ME_VERT_TMP_TAG) {
|
|
|
|
PRINT_ERR("\tPoly %u has duplicated vert reference at corner (%u)\n", i, j);
|
|
|
|
sp->invalid = true;
|
|
|
|
}
|
2014-08-19 21:09:10 +10:00
|
|
|
else {
|
|
|
|
mverts[ml->v].flag |= ME_VERT_TMP_TAG;
|
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
*v = ml->v;
|
|
|
|
}
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
if (sp->invalid)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Test all poly's loops. */
|
|
|
|
for (j = 0, ml = &mloops[sp->loopstart]; j < mp->totloop; j++, ml++) {
|
|
|
|
v1 = ml->v;
|
|
|
|
v2 = mloops[sp->loopstart + (j + 1) % mp->totloop].v;
|
|
|
|
if (!BLI_edgehash_haskey(edge_hash, v1, v2)) {
|
|
|
|
/* Edge not existing. */
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tPoly %u needs missing edge (%d, %d)\n", sp->index, v1, v2);
|
2012-03-15 20:10:07 +00:00
|
|
|
if (do_fixes)
|
2015-02-24 13:08:07 +11:00
|
|
|
recalc_flag.edges = true;
|
2012-03-15 20:10:07 +00:00
|
|
|
else
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
else if (ml->e >= totedge) {
|
|
|
|
/* Invalid edge idx.
|
|
|
|
* We already know from previous text that a valid edge exists, use it (if allowed)! */
|
2012-03-24 06:18:31 +00:00
|
|
|
if (do_fixes) {
|
2012-03-15 20:10:07 +00:00
|
|
|
int prev_e = ml->e;
|
|
|
|
ml->e = GET_INT_FROM_POINTER(BLI_edgehash_lookup(edge_hash, v1, v2));
|
2015-10-07 14:38:36 +11:00
|
|
|
fix_flag.loops_edge = true;
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tLoop %u has invalid edge reference (%d), fixed using edge %u\n",
|
2013-09-04 01:29:34 +00:00
|
|
|
sp->loopstart + j, prev_e, ml->e);
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
else {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tLoop %u has invalid edge reference (%u)\n", sp->loopstart + j, ml->e);
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-03-15 20:10:07 +00:00
|
|
|
me = &medges[ml->e];
|
|
|
|
if (IS_REMOVED_EDGE(me) || !((me->v1 == v1 && me->v2 == v2) || (me->v1 == v2 && me->v2 == v1))) {
|
|
|
|
/* The pointed edge is invalid (tagged as removed, or vert idx mismatch),
|
|
|
|
* and we already know from previous test that a valid one exists, use it (if allowed)! */
|
2012-03-24 06:18:31 +00:00
|
|
|
if (do_fixes) {
|
2012-03-15 20:10:07 +00:00
|
|
|
int prev_e = ml->e;
|
|
|
|
ml->e = GET_INT_FROM_POINTER(BLI_edgehash_lookup(edge_hash, v1, v2));
|
2015-10-07 14:38:36 +11:00
|
|
|
fix_flag.loops_edge = true;
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tPoly %u has invalid edge reference (%d), fixed using edge %u\n",
|
2013-09-04 01:29:34 +00:00
|
|
|
sp->index, prev_e, ml->e);
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
else {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tPoly %u has invalid edge reference (%u)\n", sp->index, ml->e);
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
}
|
2011-02-09 01:27:46 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
if (!sp->invalid) {
|
2015-11-15 17:11:19 +01:00
|
|
|
/* Needed for checking polys using same verts below. */
|
|
|
|
qsort(sp->verts, sp->numverts, sizeof(int), int_cmp);
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-02-10 14:13:13 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Second check pass, testing polys using the same verts. */
|
|
|
|
qsort(sort_polys, totpoly, sizeof(SortPoly), search_poly_cmp);
|
|
|
|
sp = prev_sp = sort_polys;
|
|
|
|
sp++;
|
|
|
|
|
|
|
|
for (i = 1; i < totpoly; i++, sp++) {
|
|
|
|
int p1_nv = sp->numverts, p2_nv = prev_sp->numverts;
|
2014-04-27 00:20:13 +10:00
|
|
|
const int *p1_v = sp->verts, *p2_v = prev_sp->verts;
|
2015-11-15 17:11:19 +01:00
|
|
|
|
|
|
|
if (sp->invalid) {
|
|
|
|
/* break, because all known invalid polys have been put at the end by qsort with search_poly_cmp. */
|
2012-03-15 20:10:07 +00:00
|
|
|
break;
|
2015-11-15 17:11:19 +01:00
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Test same polys. */
|
|
|
|
#if 0
|
2015-11-15 17:11:19 +01:00
|
|
|
{
|
|
|
|
bool p1_sub = true, p2_sub = true;
|
|
|
|
|
|
|
|
/* NOTE: This performs a sub-set test. */
|
|
|
|
/* XXX This (and the sort of verts list) is better than systematic
|
|
|
|
* search of all verts of one list into the other if lists have
|
|
|
|
* a fair amount of elements.
|
|
|
|
* Not sure however it's worth it in this case?
|
|
|
|
* But as we also need sorted vert list to check verts multi-used
|
|
|
|
* (in first pass of checks)... */
|
|
|
|
/* XXX If we consider only "equal" polys (i.e. using exactly same set of verts)
|
|
|
|
* as invalid, better to replace this by a simple memory cmp... */
|
|
|
|
while ((p1_nv && p2_nv) && (p1_sub || p2_sub)) {
|
|
|
|
if (*p1_v < *p2_v) {
|
|
|
|
if (p1_sub)
|
|
|
|
p1_sub = false;
|
|
|
|
p1_nv--;
|
|
|
|
p1_v++;
|
|
|
|
}
|
|
|
|
else if (*p2_v < *p1_v) {
|
|
|
|
if (p2_sub)
|
|
|
|
p2_sub = false;
|
|
|
|
p2_nv--;
|
|
|
|
p2_v++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Equality, both next verts. */
|
|
|
|
p1_nv--;
|
|
|
|
p2_nv--;
|
|
|
|
p1_v++;
|
|
|
|
p2_v++;
|
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
2015-11-15 17:11:19 +01:00
|
|
|
if (p1_nv && p1_sub)
|
|
|
|
p1_sub = false;
|
|
|
|
else if (p2_nv && p2_sub)
|
|
|
|
p2_sub = false;
|
|
|
|
|
|
|
|
if (p1_sub && p2_sub) {
|
|
|
|
PRINT("\tPolys %u and %u use same vertices, considering poly %u as invalid.\n",
|
|
|
|
prev_sp->index, sp->index, sp->index);
|
|
|
|
sp->invalid = true;
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2015-11-15 17:11:19 +01:00
|
|
|
/* XXX In fact, these might be valid? :/ */
|
|
|
|
else if (p1_sub) {
|
|
|
|
PRINT("\t%u is a sub-poly of %u, considering it as invalid.\n", sp->index, prev_sp->index);
|
|
|
|
sp->invalid = true;
|
|
|
|
}
|
|
|
|
else if (p2_sub) {
|
|
|
|
PRINT("\t%u is a sub-poly of %u, considering it as invalid.\n", prev_sp->index, sp->index);
|
|
|
|
prev_sp->invalid = true;
|
|
|
|
prev_sp = sp; /* sp is new reference poly. */
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
#else
|
2012-03-24 06:18:31 +00:00
|
|
|
if ((p1_nv == p2_nv) && (memcmp(p1_v, p2_v, p1_nv * sizeof(*p1_v)) == 0)) {
|
2012-03-15 20:10:07 +00:00
|
|
|
if (do_verbose) {
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tPolys %u and %u use same vertices (%d",
|
2013-09-04 01:29:34 +00:00
|
|
|
prev_sp->index, sp->index, *p1_v);
|
2012-03-15 20:10:07 +00:00
|
|
|
for (j = 1; j < p1_nv; j++)
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR(", %d", p1_v[j]);
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("), considering poly %u as invalid.\n", sp->index);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
is_valid = false;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
2014-04-01 11:34:00 +11:00
|
|
|
sp->invalid = true;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
else {
|
|
|
|
prev_sp = sp;
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* Third check pass, testing loops used by none or more than one poly. */
|
|
|
|
qsort(sort_polys, totpoly, sizeof(SortPoly), search_polyloop_cmp);
|
|
|
|
sp = sort_polys;
|
|
|
|
prev_sp = NULL;
|
|
|
|
prev_end = 0;
|
|
|
|
for (i = 0; i < totpoly; i++, sp++) {
|
|
|
|
/* Free this now, we don't need it anymore, and avoid us another loop! */
|
|
|
|
if (sp->verts)
|
|
|
|
MEM_freeN(sp->verts);
|
|
|
|
|
|
|
|
/* Note above prev_sp: in following code, we make sure it is always valid poly (or NULL). */
|
|
|
|
if (sp->invalid) {
|
|
|
|
if (do_fixes) {
|
|
|
|
REMOVE_POLY_TAG((&mpolys[sp->index]));
|
|
|
|
/* DO NOT REMOVE ITS LOOPS!!!
|
|
|
|
* As already invalid polys are at the end of the SortPoly list, the loops they
|
|
|
|
* were the only users have already been tagged as "to remove" during previous
|
2012-04-21 14:14:58 +00:00
|
|
|
* iterations, and we don't want to remove some loops that may be used by
|
2012-03-15 20:10:07 +00:00
|
|
|
* another valid poly! */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Test loops users. */
|
|
|
|
else {
|
|
|
|
/* Unused loops. */
|
|
|
|
if (prev_end < sp->loopstart) {
|
|
|
|
for (j = prev_end, ml = &mloops[prev_end]; j < sp->loopstart; j++, ml++) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tLoop %u is unused.\n", j);
|
2012-03-15 20:10:07 +00:00
|
|
|
if (do_fixes)
|
|
|
|
REMOVE_LOOP_TAG(ml);
|
|
|
|
}
|
|
|
|
prev_end = sp->loopstart + sp->numverts;
|
|
|
|
prev_sp = sp;
|
|
|
|
}
|
|
|
|
/* Multi-used loops. */
|
|
|
|
else if (prev_end > sp->loopstart) {
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tPolys %u and %u share loops from %d to %d, considering poly %u as invalid.\n",
|
2013-09-04 01:29:34 +00:00
|
|
|
prev_sp->index, sp->index, sp->loopstart, prev_end, sp->index);
|
2012-03-15 20:10:07 +00:00
|
|
|
if (do_fixes) {
|
|
|
|
REMOVE_POLY_TAG((&mpolys[sp->index]));
|
|
|
|
/* DO NOT REMOVE ITS LOOPS!!!
|
|
|
|
* They might be used by some next, valid poly!
|
|
|
|
* Just not updating prev_end/prev_sp vars is enough to ensure the loops
|
|
|
|
* effectively no more needed will be marked as "to be removed"! */
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2011-02-10 14:13:13 +00:00
|
|
|
else {
|
2012-03-15 20:10:07 +00:00
|
|
|
prev_end = sp->loopstart + sp->numverts;
|
|
|
|
prev_sp = sp;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2012-03-15 20:10:07 +00:00
|
|
|
/* We may have some remaining unused loops to get rid of! */
|
|
|
|
if (prev_end < totloop) {
|
|
|
|
for (j = prev_end, ml = &mloops[prev_end]; j < totloop; j++, ml++) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tLoop %u is unused.\n", j);
|
2012-03-15 20:10:07 +00:00
|
|
|
if (do_fixes)
|
|
|
|
REMOVE_LOOP_TAG(ml);
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2011-02-10 14:13:13 +00:00
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
MEM_freeN(sort_polys);
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BLI_edgehash_free(edge_hash, NULL);
|
2011-12-08 04:51:03 +00:00
|
|
|
|
|
|
|
/* fix deform verts */
|
|
|
|
if (dverts) {
|
|
|
|
MDeformVert *dv;
|
2012-04-18 09:16:30 +00:00
|
|
|
for (i = 0, dv = dverts; i < totvert; i++, dv++) {
|
2011-12-09 20:29:21 +00:00
|
|
|
MDeformWeight *dw;
|
2011-12-08 04:51:03 +00:00
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
for (j = 0, dw = dv->dw; j < dv->totweight; j++, dw++) {
|
2014-11-21 14:14:50 +01:00
|
|
|
/* note, greater than max defgroups is accounted for in our code, but not < 0 */
|
2016-05-16 00:48:02 +02:00
|
|
|
if (!isfinite(dw->weight)) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tVertex deform %u, group %d has weight: %f\n", i, dw->def_nr, dw->weight);
|
2011-12-08 04:51:03 +00:00
|
|
|
if (do_fixes) {
|
2012-04-18 09:16:30 +00:00
|
|
|
dw->weight = 0.0f;
|
2015-02-24 13:08:07 +11:00
|
|
|
fix_flag.verts_weight = true;
|
2011-12-08 04:51:03 +00:00
|
|
|
}
|
|
|
|
}
|
2011-12-09 20:29:21 +00:00
|
|
|
else if (dw->weight < 0.0f || dw->weight > 1.0f) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tVertex deform %u, group %d has weight: %f\n", i, dw->def_nr, dw->weight);
|
2011-12-09 20:29:21 +00:00
|
|
|
if (do_fixes) {
|
|
|
|
CLAMP(dw->weight, 0.0f, 1.0f);
|
2015-02-24 13:08:07 +11:00
|
|
|
fix_flag.verts_weight = true;
|
2011-12-08 04:51:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dw->def_nr < 0) {
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_ERR("\tVertex deform %u, has invalid group %d\n", i, dw->def_nr);
|
2011-12-08 04:51:03 +00:00
|
|
|
if (do_fixes) {
|
|
|
|
defvert_remove_group(dv, dw);
|
2015-02-24 13:08:07 +11:00
|
|
|
fix_flag.verts_weight = true;
|
|
|
|
|
2011-12-08 04:51:03 +00:00
|
|
|
if (dv->dw) {
|
|
|
|
/* re-allocated, the new values compensate for stepping
|
|
|
|
* within the for loop and may not be valid */
|
|
|
|
j--;
|
2012-04-18 09:16:30 +00:00
|
|
|
dw = dv->dw + j;
|
2011-12-09 07:23:17 +00:00
|
|
|
|
2011-12-08 04:51:03 +00:00
|
|
|
}
|
|
|
|
else { /* all freed */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
# undef REMOVE_EDGE_TAG
|
|
|
|
# undef IS_REMOVED_EDGE
|
|
|
|
# undef REMOVE_LOOP_TAG
|
|
|
|
# undef REMOVE_POLY_TAG
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (mesh) {
|
2015-02-24 13:08:07 +11:00
|
|
|
if (free_flag.faces) {
|
2012-06-28 09:08:11 +00:00
|
|
|
BKE_mesh_strip_loose_faces(mesh);
|
|
|
|
}
|
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
if (free_flag.polyloops) {
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_strip_loose_polysloops(mesh);
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
if (free_flag.edges) {
|
2012-05-05 21:28:12 +00:00
|
|
|
BKE_mesh_strip_loose_edges(mesh);
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
if (recalc_flag.edges) {
|
2013-03-16 01:19:03 +00:00
|
|
|
BKE_mesh_calc_edges(mesh, true, false);
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
}
|
2011-02-10 09:29:31 +00:00
|
|
|
|
2012-07-02 09:57:31 +00:00
|
|
|
if (mesh && mesh->mselect) {
|
|
|
|
MSelect *msel;
|
|
|
|
|
|
|
|
for (i = 0, msel = mesh->mselect; i < mesh->totselect; i++, msel++) {
|
2012-07-05 13:02:42 +00:00
|
|
|
int tot_elem = 0;
|
2012-07-02 09:57:31 +00:00
|
|
|
|
|
|
|
if (msel->index < 0) {
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tMesh select element %u type %d index is negative, "
|
2013-09-04 01:29:34 +00:00
|
|
|
"resetting selection stack.\n", i, msel->type);
|
2015-02-24 13:08:07 +11:00
|
|
|
free_flag.mselect = do_fixes;
|
2012-07-02 09:57:31 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (msel->type) {
|
|
|
|
case ME_VSEL:
|
|
|
|
tot_elem = mesh->totvert;
|
|
|
|
break;
|
|
|
|
case ME_ESEL:
|
|
|
|
tot_elem = mesh->totedge;
|
|
|
|
break;
|
|
|
|
case ME_FSEL:
|
|
|
|
tot_elem = mesh->totface;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msel->index > tot_elem) {
|
2015-06-19 12:29:06 +02:00
|
|
|
PRINT_ERR("\tMesh select element %u type %d index %d is larger than data array size %d, "
|
2013-09-04 01:29:34 +00:00
|
|
|
"resetting selection stack.\n", i, msel->type, msel->index, tot_elem);
|
2012-07-02 09:57:31 +00:00
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
free_flag.mselect = do_fixes;
|
2012-07-02 09:57:31 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
if (free_flag.mselect) {
|
2012-07-02 09:57:31 +00:00
|
|
|
MEM_freeN(mesh->mselect);
|
|
|
|
mesh->mselect = NULL;
|
|
|
|
mesh->totselect = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_MSG("%s: finished\n\n", __func__);
|
|
|
|
|
2015-02-24 13:08:07 +11:00
|
|
|
*r_changed = (fix_flag.as_flag || free_flag.as_flag || recalc_flag.as_flag);
|
|
|
|
|
2015-11-15 17:11:19 +01:00
|
|
|
BLI_assert((*r_changed == false) || (do_fixes == true));
|
2012-07-02 09:57:31 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
return is_valid;
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
static bool mesh_validate_customdata(CustomData *data, CustomDataMask mask,
|
|
|
|
const bool do_verbose, const bool do_fixes,
|
|
|
|
bool *r_change)
|
2011-10-23 17:52:20 +00:00
|
|
|
{
|
2013-09-04 01:29:34 +00:00
|
|
|
bool is_valid = true;
|
|
|
|
bool has_fixes = false;
|
|
|
|
int i = 0;
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_MSG("%s: Checking %d CD layers...\n", __func__, data->totlayer);
|
2012-07-05 13:02:42 +00:00
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
while (i < data->totlayer) {
|
|
|
|
CustomDataLayer *layer = &data->layers[i];
|
2013-09-04 01:29:34 +00:00
|
|
|
bool ok = true;
|
|
|
|
|
|
|
|
if (CustomData_layertype_is_singleton(layer->type)) {
|
|
|
|
const int layer_tot = CustomData_number_of_layers(data, layer->type);
|
|
|
|
if (layer_tot > 1) {
|
|
|
|
PRINT_ERR("\tCustomDataLayer type %d is a singleton, found %d in Mesh structure\n",
|
|
|
|
layer->type, layer_tot);
|
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
}
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
if (mask != 0) {
|
|
|
|
CustomDataMask layer_typemask = CD_TYPE_AS_MASK(layer->type);
|
|
|
|
if ((layer_typemask & mask) == 0) {
|
|
|
|
PRINT_ERR("\tCustomDataLayer type %d which isn't in the mask\n",
|
|
|
|
layer->type);
|
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
}
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
if (ok == false) {
|
2012-03-24 06:18:31 +00:00
|
|
|
if (do_fixes) {
|
2011-10-23 17:52:20 +00:00
|
|
|
CustomData_free_layer(data, layer->type, 0, i);
|
2013-09-04 01:29:34 +00:00
|
|
|
has_fixes = true;
|
2011-10-23 17:52:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (ok)
|
2011-10-23 17:52:20 +00:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
PRINT_MSG("%s: Finished (is_valid=%d)\n\n", __func__, (int)!has_fixes);
|
2012-07-05 13:02:42 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
*r_change = has_fixes;
|
|
|
|
|
|
|
|
return is_valid;
|
2011-10-23 17:52:20 +00:00
|
|
|
}
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
/**
|
|
|
|
* \returns is_valid.
|
|
|
|
*/
|
|
|
|
bool BKE_mesh_validate_all_customdata(CustomData *vdata, CustomData *edata,
|
|
|
|
CustomData *ldata, CustomData *pdata,
|
2013-09-04 06:50:15 +00:00
|
|
|
const bool check_meshmask,
|
2013-09-04 01:29:34 +00:00
|
|
|
const bool do_verbose, const bool do_fixes,
|
|
|
|
bool *r_change)
|
2011-10-23 17:52:20 +00:00
|
|
|
{
|
2013-09-04 01:29:34 +00:00
|
|
|
bool is_valid = true;
|
|
|
|
bool is_change_v, is_change_e, is_change_l, is_change_p;
|
2013-11-11 20:37:19 +00:00
|
|
|
int tot_texpoly, tot_uvloop, tot_vcolloop;
|
2013-09-04 06:50:15 +00:00
|
|
|
CustomDataMask mask = check_meshmask ? CD_MASK_MESH : 0;
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
is_valid &= mesh_validate_customdata(vdata, mask, do_verbose, do_fixes, &is_change_v);
|
|
|
|
is_valid &= mesh_validate_customdata(edata, mask, do_verbose, do_fixes, &is_change_e);
|
|
|
|
is_valid &= mesh_validate_customdata(ldata, mask, do_verbose, do_fixes, &is_change_l);
|
|
|
|
is_valid &= mesh_validate_customdata(pdata, mask, do_verbose, do_fixes, &is_change_p);
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2013-09-16 06:00:25 +00:00
|
|
|
tot_texpoly = CustomData_number_of_layers(pdata, CD_MTEXPOLY);
|
|
|
|
tot_uvloop = CustomData_number_of_layers(ldata, CD_MLOOPUV);
|
2013-11-11 20:37:19 +00:00
|
|
|
tot_vcolloop = CustomData_number_of_layers(ldata, CD_MLOOPCOL);
|
2013-09-16 06:00:25 +00:00
|
|
|
if (tot_texpoly != tot_uvloop) {
|
|
|
|
PRINT_ERR("\tCustomDataLayer mismatch, tot_texpoly(%d), tot_uvloop(%d)\n",
|
|
|
|
tot_texpoly, tot_uvloop);
|
|
|
|
}
|
2013-11-11 20:37:19 +00:00
|
|
|
if (tot_texpoly > MAX_MTFACE) {
|
|
|
|
PRINT_ERR("\tMore UV layers than %d allowed, %d last ones won't be available for render, shaders, etc.\n",
|
|
|
|
MAX_MTFACE, tot_texpoly - MAX_MTFACE);
|
|
|
|
}
|
|
|
|
if (tot_uvloop > MAX_MTFACE) {
|
|
|
|
PRINT_ERR("\tMore UV layers than %d allowed, %d last ones won't be available for render, shaders, etc.\n",
|
|
|
|
MAX_MTFACE, tot_uvloop - MAX_MTFACE);
|
|
|
|
}
|
|
|
|
if (tot_vcolloop > MAX_MCOL) {
|
|
|
|
PRINT_ERR("\tMore VCol layers than %d allowed, %d last ones won't be available for render, shaders, etc.\n",
|
|
|
|
MAX_MCOL, tot_vcolloop - MAX_MCOL);
|
|
|
|
}
|
2013-09-16 06:00:25 +00:00
|
|
|
|
2015-08-20 10:35:14 +03:00
|
|
|
/* check indices of clone/stencil */
|
|
|
|
if (do_fixes && CustomData_get_clone_layer(pdata, CD_MTEXPOLY) >= tot_texpoly) {
|
|
|
|
CustomData_set_layer_clone(pdata, CD_MTEXPOLY, 0);
|
|
|
|
is_change_p = true;
|
|
|
|
}
|
|
|
|
if (do_fixes && CustomData_get_clone_layer(ldata, CD_MLOOPUV) >= tot_uvloop) {
|
|
|
|
CustomData_set_layer_clone(ldata, CD_MLOOPUV, 0);
|
|
|
|
is_change_l = true;
|
|
|
|
}
|
|
|
|
if (do_fixes && CustomData_get_stencil_layer(pdata, CD_MTEXPOLY) >= tot_texpoly) {
|
|
|
|
CustomData_set_layer_stencil(pdata, CD_MTEXPOLY, 0);
|
|
|
|
is_change_p = true;
|
|
|
|
}
|
|
|
|
if (do_fixes && CustomData_get_stencil_layer(ldata, CD_MLOOPUV) >= tot_uvloop) {
|
|
|
|
CustomData_set_layer_stencil(ldata, CD_MLOOPUV, 0);
|
|
|
|
is_change_l = true;
|
|
|
|
}
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
*r_change = (is_change_v || is_change_e || is_change_l || is_change_p);
|
|
|
|
|
|
|
|
return is_valid;
|
2011-10-23 17:52:20 +00:00
|
|
|
}
|
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
/**
|
|
|
|
* \see #DM_is_valid to call on derived meshes
|
|
|
|
*
|
|
|
|
* \returns true if a change is made.
|
|
|
|
*/
|
2015-02-05 14:03:01 +01:00
|
|
|
int BKE_mesh_validate(Mesh *me, const int do_verbose, const int cddata_check_mask)
|
2011-02-09 01:27:46 +00:00
|
|
|
{
|
2013-09-04 01:29:34 +00:00
|
|
|
bool is_valid = true;
|
2013-11-26 06:39:14 +11:00
|
|
|
bool changed;
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (do_verbose) {
|
2012-04-18 09:16:30 +00:00
|
|
|
printf("MESH: %s\n", me->id.name + 2);
|
2011-04-25 06:44:43 +00:00
|
|
|
}
|
2011-10-23 17:52:20 +00:00
|
|
|
|
2013-09-04 01:29:34 +00:00
|
|
|
is_valid &= BKE_mesh_validate_all_customdata(
|
|
|
|
&me->vdata, &me->edata, &me->ldata, &me->pdata,
|
2015-02-05 14:03:01 +01:00
|
|
|
cddata_check_mask,
|
2013-09-04 01:29:34 +00:00
|
|
|
do_verbose, true,
|
2013-11-26 06:39:14 +11:00
|
|
|
&changed);
|
2013-09-04 01:29:34 +00:00
|
|
|
|
|
|
|
is_valid &= BKE_mesh_validate_arrays(
|
|
|
|
me,
|
|
|
|
me->mvert, me->totvert,
|
|
|
|
me->medge, me->totedge,
|
|
|
|
me->mface, me->totface,
|
|
|
|
me->mloop, me->totloop,
|
|
|
|
me->mpoly, me->totpoly,
|
|
|
|
me->dvert,
|
|
|
|
do_verbose, true,
|
2013-11-26 06:39:14 +11:00
|
|
|
&changed);
|
2013-09-04 01:29:34 +00:00
|
|
|
|
2013-11-26 06:39:14 +11:00
|
|
|
if (changed) {
|
2017-04-06 16:11:50 +02:00
|
|
|
DEG_id_tag_update(&me->id, OB_RECALC_DATA);
|
2013-03-17 19:55:10 +00:00
|
|
|
return true;
|
2012-03-15 20:10:07 +00:00
|
|
|
}
|
2013-09-04 01:29:34 +00:00
|
|
|
else {
|
|
|
|
return false;
|
|
|
|
}
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2013-09-16 06:00:25 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Duplicate of BM_mesh_cd_validate() for Mesh data.
|
|
|
|
*/
|
|
|
|
void BKE_mesh_cd_validate(Mesh *me)
|
|
|
|
{
|
|
|
|
int totlayer_mtex = CustomData_number_of_layers(&me->pdata, CD_MTEXPOLY);
|
|
|
|
int totlayer_uv = CustomData_number_of_layers(&me->ldata, CD_MLOOPUV);
|
2013-11-11 20:37:19 +00:00
|
|
|
int totlayer_mcol = CustomData_number_of_layers(&me->ldata, CD_MLOOPCOL);
|
2013-09-20 11:14:08 +00:00
|
|
|
int mtex_index = CustomData_get_layer_index(&me->pdata, CD_MTEXPOLY);
|
|
|
|
int uv_index = CustomData_get_layer_index(&me->ldata, CD_MLOOPUV);
|
|
|
|
int i;
|
2013-09-16 06:00:25 +00:00
|
|
|
|
2013-11-11 20:37:19 +00:00
|
|
|
/* XXX For now, do not delete those, just warn they are not really usable. */
|
|
|
|
if (UNLIKELY(totlayer_mtex > MAX_MTFACE)) {
|
|
|
|
printf("WARNING! More UV layers than %d allowed, %d last ones won't be available for render, shaders, etc.\n",
|
|
|
|
MAX_MTFACE, totlayer_mtex - MAX_MTFACE);
|
|
|
|
}
|
|
|
|
if (UNLIKELY(totlayer_uv > MAX_MTFACE)) {
|
|
|
|
printf("WARNING! More UV layers than %d allowed, %d last ones won't be available for render, shaders, etc.\n",
|
|
|
|
MAX_MTFACE, totlayer_uv - MAX_MTFACE);
|
|
|
|
}
|
|
|
|
if (UNLIKELY(totlayer_mcol > MAX_MCOL)) {
|
|
|
|
printf("WARNING! More VCol layers than %d allowed, %d last ones won't be available for render, shaders, etc.\n",
|
|
|
|
MAX_MCOL, totlayer_mcol - MAX_MCOL);
|
|
|
|
}
|
|
|
|
|
2013-09-16 06:00:25 +00:00
|
|
|
if (LIKELY(totlayer_mtex == totlayer_uv)) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if (totlayer_mtex < totlayer_uv) {
|
|
|
|
do {
|
2013-09-20 11:14:08 +00:00
|
|
|
const char *from_name = me->ldata.layers[uv_index + totlayer_mtex].name;
|
2013-09-16 06:00:25 +00:00
|
|
|
CustomData_add_layer_named(&me->pdata, CD_MTEXPOLY, CD_DEFAULT, NULL, me->totpoly, from_name);
|
|
|
|
CustomData_set_layer_unique_name(&me->pdata, totlayer_mtex);
|
|
|
|
} while (totlayer_uv != ++totlayer_mtex);
|
2013-09-25 11:11:41 +00:00
|
|
|
mtex_index = CustomData_get_layer_index(&me->pdata, CD_MTEXPOLY);
|
2013-09-16 06:00:25 +00:00
|
|
|
}
|
|
|
|
else if (totlayer_uv < totlayer_mtex) {
|
|
|
|
do {
|
2013-09-20 11:14:08 +00:00
|
|
|
const char *from_name = me->pdata.layers[mtex_index + totlayer_uv].name;
|
2013-09-16 06:00:25 +00:00
|
|
|
CustomData_add_layer_named(&me->ldata, CD_MLOOPUV, CD_DEFAULT, NULL, me->totloop, from_name);
|
|
|
|
CustomData_set_layer_unique_name(&me->ldata, totlayer_uv);
|
|
|
|
} while (totlayer_mtex != ++totlayer_uv);
|
2013-09-25 11:11:41 +00:00
|
|
|
uv_index = CustomData_get_layer_index(&me->ldata, CD_MLOOPUV);
|
2013-09-16 06:00:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BLI_assert(totlayer_mtex == totlayer_uv);
|
2013-09-20 11:14:08 +00:00
|
|
|
|
|
|
|
/* Check uv/tex names match as well!!! */
|
|
|
|
for (i = 0; i < totlayer_mtex; i++, mtex_index++, uv_index++) {
|
2013-09-25 11:11:41 +00:00
|
|
|
const char *name_src = me->pdata.layers[mtex_index].name;
|
|
|
|
const char *name_dst = me->ldata.layers[uv_index].name;
|
|
|
|
if (!STREQ(name_src, name_dst)) {
|
|
|
|
BKE_mesh_uv_cdlayer_rename_index(me, mtex_index, uv_index, -1, name_src, false);
|
2013-09-20 11:14:08 +00:00
|
|
|
}
|
|
|
|
}
|
2013-09-16 06:00:25 +00:00
|
|
|
}
|
2013-09-09 02:11:44 +00:00
|
|
|
|
2014-07-17 17:12:12 +02:00
|
|
|
/**
|
|
|
|
* Check all material indices of polygons are valid, invalid ones are set to 0.
|
|
|
|
* \returns is_valid.
|
|
|
|
*/
|
|
|
|
int BKE_mesh_validate_material_indices(Mesh *me)
|
|
|
|
{
|
|
|
|
MPoly *mp;
|
|
|
|
const int max_idx = max_ii(0, me->totcol - 1);
|
|
|
|
const int totpoly = me->totpoly;
|
|
|
|
int i;
|
|
|
|
bool is_valid = true;
|
|
|
|
|
|
|
|
for (mp = me->mpoly, i = 0; i < totpoly; i++, mp++) {
|
|
|
|
if (mp->mat_nr > max_idx) {
|
|
|
|
mp->mat_nr = 0;
|
|
|
|
is_valid = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_valid) {
|
2017-04-06 16:11:50 +02:00
|
|
|
DEG_id_tag_update(&me->id, OB_RECALC_DATA);
|
2014-07-17 17:12:12 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** \} */
|
2013-09-09 02:11:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/** \name Mesh Stripping (removing invalid data)
|
|
|
|
* \{ */
|
|
|
|
|
|
|
|
/* We need to keep this for edge creation (for now?), and some old readfile code... */
|
|
|
|
void BKE_mesh_strip_loose_faces(Mesh *me)
|
|
|
|
{
|
|
|
|
MFace *f;
|
|
|
|
int a, b;
|
|
|
|
|
|
|
|
for (a = b = 0, f = me->mface; a < me->totface; a++, f++) {
|
|
|
|
if (f->v3) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->mface[b], f, sizeof(me->mface[b]));
|
|
|
|
CustomData_copy_data(&me->fdata, &me->fdata, a, b, 1);
|
|
|
|
}
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->fdata, b, a - b);
|
|
|
|
me->totface = b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Works on both loops and polys! */
|
|
|
|
/* Note: It won't try to guess which loops of an invalid poly to remove!
|
|
|
|
* this is the work of the caller, to mark those loops...
|
|
|
|
* See e.g. BKE_mesh_validate_arrays(). */
|
|
|
|
void BKE_mesh_strip_loose_polysloops(Mesh *me)
|
|
|
|
{
|
|
|
|
MPoly *p;
|
|
|
|
MLoop *l;
|
|
|
|
int a, b;
|
|
|
|
/* New loops idx! */
|
|
|
|
int *new_idx = MEM_mallocN(sizeof(int) * me->totloop, __func__);
|
|
|
|
|
|
|
|
for (a = b = 0, p = me->mpoly; a < me->totpoly; a++, p++) {
|
2014-04-01 11:34:00 +11:00
|
|
|
bool invalid = false;
|
2013-09-09 02:11:44 +00:00
|
|
|
int i = p->loopstart;
|
|
|
|
int stop = i + p->totloop;
|
|
|
|
|
|
|
|
if (stop > me->totloop || stop < i) {
|
2014-04-01 11:34:00 +11:00
|
|
|
invalid = true;
|
2013-09-09 02:11:44 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
l = &me->mloop[i];
|
|
|
|
i = stop - i;
|
|
|
|
/* If one of the poly's loops is invalid, the whole poly is invalid! */
|
|
|
|
for (; i--; l++) {
|
|
|
|
if (l->e == INVALID_LOOP_EDGE_MARKER) {
|
2014-04-01 11:34:00 +11:00
|
|
|
invalid = true;
|
2013-09-09 02:11:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p->totloop >= 3 && !invalid) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->mpoly[b], p, sizeof(me->mpoly[b]));
|
|
|
|
CustomData_copy_data(&me->pdata, &me->pdata, a, b, 1);
|
|
|
|
}
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->pdata, b, a - b);
|
|
|
|
me->totpoly = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And now, get rid of invalid loops. */
|
|
|
|
for (a = b = 0, l = me->mloop; a < me->totloop; a++, l++) {
|
|
|
|
if (l->e != INVALID_LOOP_EDGE_MARKER) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->mloop[b], l, sizeof(me->mloop[b]));
|
|
|
|
CustomData_copy_data(&me->ldata, &me->ldata, a, b, 1);
|
|
|
|
}
|
|
|
|
new_idx[a] = b;
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* XXX Theoretically, we should be able to not do this, as no remaining poly
|
|
|
|
* should use any stripped loop. But for security's sake... */
|
|
|
|
new_idx[a] = -a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->ldata, b, a - b);
|
|
|
|
me->totloop = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And now, update polys' start loop index. */
|
|
|
|
/* Note: At this point, there should never be any poly using a striped loop! */
|
|
|
|
for (a = 0, p = me->mpoly; a < me->totpoly; a++, p++) {
|
|
|
|
p->loopstart = new_idx[p->loopstart];
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(new_idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void BKE_mesh_strip_loose_edges(Mesh *me)
|
|
|
|
{
|
|
|
|
MEdge *e;
|
|
|
|
MLoop *l;
|
|
|
|
int a, b;
|
|
|
|
unsigned int *new_idx = MEM_mallocN(sizeof(int) * me->totedge, __func__);
|
|
|
|
|
|
|
|
for (a = b = 0, e = me->medge; a < me->totedge; a++, e++) {
|
|
|
|
if (e->v1 != e->v2) {
|
|
|
|
if (a != b) {
|
|
|
|
memcpy(&me->medge[b], e, sizeof(me->medge[b]));
|
|
|
|
CustomData_copy_data(&me->edata, &me->edata, a, b, 1);
|
|
|
|
}
|
|
|
|
new_idx[a] = b;
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
new_idx[a] = INVALID_LOOP_EDGE_MARKER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (a != b) {
|
|
|
|
CustomData_free_elem(&me->edata, b, a - b);
|
|
|
|
me->totedge = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And now, update loops' edge indices. */
|
|
|
|
/* XXX We hope no loop was pointing to a striped edge!
|
|
|
|
* Else, its e will be set to INVALID_LOOP_EDGE_MARKER :/ */
|
|
|
|
for (a = 0, l = me->mloop; a < me->totloop; a++, l++) {
|
|
|
|
l->e = new_idx[l->e];
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(new_idx);
|
|
|
|
}
|
|
|
|
/** \} */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/** \name Mesh Edge Calculation
|
|
|
|
* \{ */
|
|
|
|
|
|
|
|
/* make edges in a Mesh, for outside of editmode */
|
|
|
|
|
|
|
|
struct EdgeSort {
|
|
|
|
unsigned int v1, v2;
|
|
|
|
char is_loose, is_draw;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* edges have to be added with lowest index first for sorting */
|
|
|
|
static void to_edgesort(struct EdgeSort *ed,
|
|
|
|
unsigned int v1, unsigned int v2,
|
|
|
|
char is_loose, short is_draw)
|
|
|
|
{
|
|
|
|
if (v1 < v2) {
|
|
|
|
ed->v1 = v1; ed->v2 = v2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ed->v1 = v2; ed->v2 = v1;
|
|
|
|
}
|
|
|
|
ed->is_loose = is_loose;
|
|
|
|
ed->is_draw = is_draw;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vergedgesort(const void *v1, const void *v2)
|
|
|
|
{
|
|
|
|
const struct EdgeSort *x1 = v1, *x2 = v2;
|
|
|
|
|
|
|
|
if (x1->v1 > x2->v1) return 1;
|
|
|
|
else if (x1->v1 < x2->v1) return -1;
|
|
|
|
else if (x1->v2 > x2->v2) return 1;
|
|
|
|
else if (x1->v2 < x2->v2) return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Create edges based on known verts and faces,
|
|
|
|
* this function is only used when loading very old blend files */
|
|
|
|
|
|
|
|
static void mesh_calc_edges_mdata(
|
|
|
|
MVert *UNUSED(allvert), MFace *allface, MLoop *allloop,
|
|
|
|
MPoly *allpoly, int UNUSED(totvert), int totface, int UNUSED(totloop), int totpoly,
|
|
|
|
const bool use_old,
|
|
|
|
MEdge **r_medge, int *r_totedge)
|
|
|
|
{
|
|
|
|
MPoly *mpoly;
|
|
|
|
MFace *mface;
|
|
|
|
MEdge *medge, *med;
|
|
|
|
EdgeHash *hash;
|
|
|
|
struct EdgeSort *edsort, *ed;
|
|
|
|
int a, totedge = 0;
|
|
|
|
unsigned int totedge_final = 0;
|
|
|
|
unsigned int edge_index;
|
|
|
|
|
|
|
|
/* we put all edges in array, sort them, and detect doubles that way */
|
|
|
|
|
|
|
|
for (a = totface, mface = allface; a > 0; a--, mface++) {
|
|
|
|
if (mface->v4) totedge += 4;
|
|
|
|
else if (mface->v3) totedge += 3;
|
|
|
|
else totedge += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (totedge == 0) {
|
|
|
|
/* flag that mesh has edges */
|
|
|
|
(*r_medge) = MEM_callocN(0, __func__);
|
|
|
|
(*r_totedge) = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ed = edsort = MEM_mallocN(totedge * sizeof(struct EdgeSort), "EdgeSort");
|
|
|
|
|
|
|
|
for (a = totface, mface = allface; a > 0; a--, mface++) {
|
|
|
|
to_edgesort(ed++, mface->v1, mface->v2, !mface->v3, mface->edcode & ME_V1V2);
|
|
|
|
if (mface->v4) {
|
|
|
|
to_edgesort(ed++, mface->v2, mface->v3, 0, mface->edcode & ME_V2V3);
|
|
|
|
to_edgesort(ed++, mface->v3, mface->v4, 0, mface->edcode & ME_V3V4);
|
|
|
|
to_edgesort(ed++, mface->v4, mface->v1, 0, mface->edcode & ME_V4V1);
|
|
|
|
}
|
|
|
|
else if (mface->v3) {
|
|
|
|
to_edgesort(ed++, mface->v2, mface->v3, 0, mface->edcode & ME_V2V3);
|
|
|
|
to_edgesort(ed++, mface->v3, mface->v1, 0, mface->edcode & ME_V3V1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qsort(edsort, totedge, sizeof(struct EdgeSort), vergedgesort);
|
|
|
|
|
|
|
|
/* count final amount */
|
|
|
|
for (a = totedge, ed = edsort; a > 1; a--, ed++) {
|
|
|
|
/* edge is unique when it differs from next edge, or is last */
|
|
|
|
if (ed->v1 != (ed + 1)->v1 || ed->v2 != (ed + 1)->v2) totedge_final++;
|
|
|
|
}
|
|
|
|
totedge_final++;
|
|
|
|
|
|
|
|
medge = MEM_callocN(sizeof(MEdge) * totedge_final, __func__);
|
|
|
|
|
|
|
|
for (a = totedge, med = medge, ed = edsort; a > 1; a--, ed++) {
|
|
|
|
/* edge is unique when it differs from next edge, or is last */
|
|
|
|
if (ed->v1 != (ed + 1)->v1 || ed->v2 != (ed + 1)->v2) {
|
|
|
|
med->v1 = ed->v1;
|
|
|
|
med->v2 = ed->v2;
|
|
|
|
if (use_old == false || ed->is_draw) med->flag = ME_EDGEDRAW | ME_EDGERENDER;
|
|
|
|
if (ed->is_loose) med->flag |= ME_LOOSEEDGE;
|
|
|
|
|
|
|
|
/* order is swapped so extruding this edge as a surface wont flip face normals
|
|
|
|
* with cyclic curves */
|
|
|
|
if (ed->v1 + 1 != ed->v2) {
|
|
|
|
SWAP(unsigned int, med->v1, med->v2);
|
|
|
|
}
|
|
|
|
med++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* equal edge, we merge the drawflag */
|
|
|
|
(ed + 1)->is_draw |= ed->is_draw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* last edge */
|
|
|
|
med->v1 = ed->v1;
|
|
|
|
med->v2 = ed->v2;
|
|
|
|
med->flag = ME_EDGEDRAW;
|
|
|
|
if (ed->is_loose) med->flag |= ME_LOOSEEDGE;
|
|
|
|
med->flag |= ME_EDGERENDER;
|
|
|
|
|
|
|
|
MEM_freeN(edsort);
|
|
|
|
|
|
|
|
/* set edge members of mloops */
|
|
|
|
hash = BLI_edgehash_new_ex(__func__, totedge_final);
|
|
|
|
for (edge_index = 0, med = medge; edge_index < totedge_final; edge_index++, med++) {
|
|
|
|
BLI_edgehash_insert(hash, med->v1, med->v2, SET_UINT_IN_POINTER(edge_index));
|
|
|
|
}
|
|
|
|
|
|
|
|
mpoly = allpoly;
|
|
|
|
for (a = 0; a < totpoly; a++, mpoly++) {
|
|
|
|
MLoop *ml, *ml_next;
|
|
|
|
int i = mpoly->totloop;
|
|
|
|
|
|
|
|
ml_next = allloop + mpoly->loopstart; /* first loop */
|
|
|
|
ml = &ml_next[i - 1]; /* last loop */
|
|
|
|
|
|
|
|
while (i-- != 0) {
|
|
|
|
ml->e = GET_UINT_FROM_POINTER(BLI_edgehash_lookup(hash, ml->v, ml_next->v));
|
|
|
|
ml = ml_next;
|
|
|
|
ml_next++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_edgehash_free(hash, NULL);
|
|
|
|
|
|
|
|
*r_medge = medge;
|
|
|
|
*r_totedge = totedge_final;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If the mesh is from a very old blender version,
|
|
|
|
* convert mface->edcode to edge drawflags
|
|
|
|
*/
|
|
|
|
void BKE_mesh_calc_edges_legacy(Mesh *me, const bool use_old)
|
|
|
|
{
|
|
|
|
MEdge *medge;
|
|
|
|
int totedge = 0;
|
|
|
|
|
|
|
|
mesh_calc_edges_mdata(me->mvert, me->mface, me->mloop, me->mpoly,
|
|
|
|
me->totvert, me->totface, me->totloop, me->totpoly,
|
|
|
|
use_old, &medge, &totedge);
|
|
|
|
|
|
|
|
if (totedge == 0) {
|
|
|
|
/* flag that mesh has edges */
|
|
|
|
me->medge = medge;
|
|
|
|
me->totedge = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
medge = CustomData_add_layer(&me->edata, CD_MEDGE, CD_ASSIGN, medge, totedge);
|
|
|
|
me->medge = medge;
|
|
|
|
me->totedge = totedge;
|
|
|
|
|
|
|
|
BKE_mesh_strip_loose_faces(me);
|
|
|
|
}
|
|
|
|
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2013-03-01 14:47:06 +00:00
|
|
|
/**
|
|
|
|
* Calculate edges from polygons
|
|
|
|
*
|
|
|
|
* \param mesh The mesh to add edges into
|
|
|
|
* \param update When true create new edges co-exist
|
|
|
|
*/
|
2013-03-16 01:19:03 +00:00
|
|
|
void BKE_mesh_calc_edges(Mesh *mesh, bool update, const bool select)
|
2011-02-09 15:13:20 +00:00
|
|
|
{
|
|
|
|
CustomData edata;
|
|
|
|
EdgeHashIterator *ehi;
|
2012-10-12 14:35:10 +00:00
|
|
|
MPoly *mp;
|
2011-02-09 15:13:20 +00:00
|
|
|
MEdge *med, *med_orig;
|
2013-08-24 13:47:57 +00:00
|
|
|
EdgeHash *eh;
|
|
|
|
unsigned int eh_reserve;
|
2012-03-15 20:10:07 +00:00
|
|
|
int i, totedge, totpoly = mesh->totpoly;
|
2011-10-09 16:59:48 +00:00
|
|
|
int med_index;
|
2013-03-16 01:19:03 +00:00
|
|
|
/* select for newly created meshes which are selected [#25595] */
|
|
|
|
const short ed_flag = (ME_EDGEDRAW | ME_EDGERENDER) | (select ? SELECT : 0);
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2012-04-18 09:16:30 +00:00
|
|
|
if (mesh->totedge == 0)
|
2013-03-16 01:19:03 +00:00
|
|
|
update = false;
|
2011-02-09 15:13:20 +00:00
|
|
|
|
2013-08-24 13:47:57 +00:00
|
|
|
eh_reserve = max_ii(update ? mesh->totedge : 0, BLI_EDGEHASH_SIZE_GUESS_FROM_POLYS(totpoly));
|
|
|
|
eh = BLI_edgehash_new_ex(__func__, eh_reserve);
|
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (update) {
|
2011-02-09 15:13:20 +00:00
|
|
|
/* assume existing edges are valid
|
|
|
|
* useful when adding more faces and generating edges from them */
|
2012-04-18 09:16:30 +00:00
|
|
|
med = mesh->medge;
|
|
|
|
for (i = 0; i < mesh->totedge; i++, med++)
|
2011-02-09 15:13:20 +00:00
|
|
|
BLI_edgehash_insert(eh, med->v1, med->v2, med);
|
|
|
|
}
|
|
|
|
|
2012-03-15 20:10:07 +00:00
|
|
|
/* mesh loops (bmesh only) */
|
2012-10-12 14:35:10 +00:00
|
|
|
for (mp = mesh->mpoly, i = 0; i < totpoly; mp++, i++) {
|
2012-04-18 09:16:30 +00:00
|
|
|
MLoop *l = &mesh->mloop[mp->loopstart];
|
2015-03-03 15:32:35 +11:00
|
|
|
int j, v_prev = (l + (mp->totloop - 1))->v;
|
2012-04-18 09:16:30 +00:00
|
|
|
for (j = 0; j < mp->totloop; j++, l++) {
|
2015-03-03 15:32:35 +11:00
|
|
|
if (v_prev != l->v) {
|
2015-04-07 10:53:58 +10:00
|
|
|
void **val_p;
|
|
|
|
if (!BLI_edgehash_ensure_p(eh, v_prev, l->v, &val_p)) {
|
|
|
|
*val_p = NULL;
|
2015-03-03 15:32:35 +11:00
|
|
|
}
|
2011-09-01 08:27:35 +00:00
|
|
|
}
|
2015-03-03 15:32:35 +11:00
|
|
|
v_prev = l->v;
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
totedge = BLI_edgehash_size(eh);
|
|
|
|
|
|
|
|
/* write new edges into a temporary CustomData */
|
2012-10-31 17:03:31 +00:00
|
|
|
CustomData_reset(&edata);
|
2011-02-09 15:13:20 +00:00
|
|
|
CustomData_add_layer(&edata, CD_MEDGE, CD_CALLOC, NULL, totedge);
|
|
|
|
|
|
|
|
med = CustomData_get_layer(&edata, CD_MEDGE);
|
2012-05-20 19:49:27 +00:00
|
|
|
for (ehi = BLI_edgehashIterator_new(eh), i = 0;
|
2014-04-01 11:34:00 +11:00
|
|
|
BLI_edgehashIterator_isDone(ehi) == false;
|
2012-05-20 19:49:27 +00:00
|
|
|
BLI_edgehashIterator_step(ehi), ++i, ++med)
|
|
|
|
{
|
2012-04-18 09:16:30 +00:00
|
|
|
if (update && (med_orig = BLI_edgehashIterator_getValue(ehi))) {
|
|
|
|
*med = *med_orig; /* copy from the original */
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
else {
|
2011-12-28 10:06:10 +00:00
|
|
|
BLI_edgehashIterator_getKey(ehi, &med->v1, &med->v2);
|
2013-03-16 01:19:03 +00:00
|
|
|
med->flag = ed_flag;
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
2011-10-09 16:59:48 +00:00
|
|
|
|
|
|
|
/* store the new edge index in the hash value */
|
|
|
|
BLI_edgehashIterator_setValue(ehi, SET_INT_IN_POINTER(i));
|
2011-02-09 15:13:20 +00:00
|
|
|
}
|
|
|
|
BLI_edgehashIterator_free(ehi);
|
|
|
|
|
2011-10-09 16:59:48 +00:00
|
|
|
if (mesh->totpoly) {
|
|
|
|
/* second pass, iterate through all loops again and assign
|
2012-03-03 20:19:11 +00:00
|
|
|
* the newly created edges to them. */
|
2012-10-12 14:35:10 +00:00
|
|
|
for (mp = mesh->mpoly, i = 0; i < mesh->totpoly; mp++, i++) {
|
2012-04-18 09:16:30 +00:00
|
|
|
MLoop *l = &mesh->mloop[mp->loopstart];
|
|
|
|
MLoop *l_prev = (l + (mp->totloop - 1));
|
2011-10-09 16:59:48 +00:00
|
|
|
int j;
|
2012-04-18 09:16:30 +00:00
|
|
|
for (j = 0; j < mp->totloop; j++, l++) {
|
2011-10-09 16:59:48 +00:00
|
|
|
/* lookup hashed edge index */
|
2011-10-10 14:56:09 +00:00
|
|
|
med_index = GET_INT_FROM_POINTER(BLI_edgehash_lookup(eh, l_prev->v, l->v));
|
2011-10-09 16:59:48 +00:00
|
|
|
l_prev->e = med_index;
|
2012-04-18 09:16:30 +00:00
|
|
|
l_prev = l;
|
2011-10-09 16:59:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-09 15:13:20 +00:00
|
|
|
/* free old CustomData and assign new one */
|
|
|
|
CustomData_free(&mesh->edata, mesh->totedge);
|
|
|
|
mesh->edata = edata;
|
|
|
|
mesh->totedge = totedge;
|
|
|
|
|
|
|
|
mesh->medge = CustomData_get_layer(&mesh->edata, CD_MEDGE);
|
|
|
|
|
|
|
|
BLI_edgehash_free(eh, NULL);
|
2011-02-09 01:27:46 +00:00
|
|
|
}
|
2013-09-09 02:11:44 +00:00
|
|
|
/** \} */
|