| 
									
										
										
										
											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"
 | 
					
						
							|  |  |  | #include "BKE_depsgraph.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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 	int invalid; /* Poly index. */ | 
					
						
							|  |  |  | } 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; | 
					
						
							|  |  |  | 	int idx = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* 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 first non-egal verts (remember verts of valid polys are sorted). */ | 
					
						
							|  |  |  | 	while (idx < max_idx && sp1->verts[idx] == sp2->verts[idx]) | 
					
						
							|  |  |  | 		idx++; | 
					
						
							|  |  |  | 	return sp1->verts[idx] > sp2->verts[idx] ? 1 : sp1->verts[idx] < sp2->verts[idx] ? -1 : | 
					
						
							|  |  |  | 	       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, | 
					
						
							|  |  |  |                               bool *r_change) | 
					
						
							| 
									
										
										
										
											2011-02-09 01:27:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | #   define REMOVE_EDGE_TAG(_me) { _me->v2 = _me->v1; do_edge_free = true; } (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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | #   define REMOVE_LOOP_TAG(_ml) { _ml->e = INVALID_LOOP_EDGE_MARKER; do_polyloop_free = true; } (void)0
 | 
					
						
							|  |  |  | #   define REMOVE_POLY_TAG(_mp) { _mp->totloop *= -1; do_polyloop_free = true; } (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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | 	bool do_edge_free = false; | 
					
						
							|  |  |  | 	bool do_face_free = false; | 
					
						
							|  |  |  | 	bool do_polyloop_free = false; /* This regroups loops and polys! */ | 
					
						
							| 
									
										
										
										
											2011-12-09 07:23:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | 	bool verts_fixed = false; | 
					
						
							|  |  |  | 	bool vert_weights_fixed = false; | 
					
						
							|  |  |  | 	bool msel_fixed = false; | 
					
						
							| 
									
										
										
										
											2011-02-09 15:13:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | 	bool do_edge_recalc = false; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 		do_edge_recalc = do_fixes; | 
					
						
							| 
									
										
										
										
											2011-02-09 01:27:46 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-18 09:16:30 +00:00
										 |  |  | 	for (i = 1; i < totvert; i++, mv++) { | 
					
						
							|  |  |  | 		int 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++) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (!finite(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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-18 09:16:30 +00:00
										 |  |  | 					verts_fixed = 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) | 
					
						
							|  |  |  | 				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; | 
					
						
							|  |  |  | 				verts_fixed = 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++) { | 
					
						
							|  |  |  | 		int remove = FALSE; | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (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
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (remove == FALSE) { | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 			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) { | 
					
						
							|  |  |  | #		define REMOVE_FACE_TAG(_mf) { _mf->v3 = 0; do_face_free = TRUE; } (void)0
 | 
					
						
							|  |  |  | #		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); \ | 
					
						
							| 
									
										
										
										
											2012-06-28 09:08:11 +00:00
										 |  |  | 						do_edge_recalc = TRUE; \ | 
					
						
							| 
									
										
										
										
											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++) { | 
					
						
							|  |  |  | 			int remove = FALSE; | 
					
						
							|  |  |  | 			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--); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (remove == FALSE) { | 
					
						
							|  |  |  | 				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); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (remove == FALSE) { | 
					
						
							|  |  |  | 					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++) { | 
					
						
							|  |  |  | 			int remove = FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* 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
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 	/* Checking loops and polys is a bit tricky, as they are quite intricated...
 | 
					
						
							|  |  |  | 	 * | 
					
						
							|  |  |  | 	 * 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; | 
					
						
							|  |  |  | 		for (i = 0, mp = mpolys; i < totpoly; i++, mp++, sp++) { | 
					
						
							|  |  |  | 			sp->index = i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (mp->loopstart < 0 || mp->totloop < 3) { | 
					
						
							|  |  |  | 				/* Invalid loop data. */ | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 				PRINT_ERR("\tPoly %u is invalid (loopstart: %u, totloop: %u)\n", | 
					
						
							|  |  |  | 				          sp->index, mp->loopstart, mp->totloop); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00: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. */ | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 				PRINT_ERR("\tPoly %u uses loops out of range (loopstart: %u, loopend: %u, max nbr of loops: %u)\n", | 
					
						
							|  |  |  | 				          sp->index, mp->loopstart, mp->loopstart + mp->totloop - 1, totloop - 1); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00: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. */ | 
					
						
							|  |  |  | 				sp->invalid = FALSE; | 
					
						
							|  |  |  | 				sp->verts = v = MEM_mallocN(sizeof(int) * mp->totloop, "Vert idx of SortPoly"); | 
					
						
							|  |  |  | 				sp->numverts = mp->totloop; | 
					
						
							|  |  |  | 				sp->loopstart = mp->loopstart; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* 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); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 						sp->invalid = TRUE; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-04-25 07:23:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-07 15:09:06 +00:00
										 |  |  | 				/* is the same vertex used more than once */ | 
					
						
							| 
									
										
										
										
											2012-04-25 07:23:20 +00:00
										 |  |  | 				if (!sp->invalid) { | 
					
						
							|  |  |  | 					v = sp->verts; | 
					
						
							|  |  |  | 					for (j = 0; j < mp->totloop; j++, v++) { | 
					
						
							|  |  |  | 						if ((mverts[*v].flag & ME_VERT_TMP_TAG) == 0) { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 							PRINT_ERR("\tPoly %u has duplicate vert reference at corner (%u)\n", i, j); | 
					
						
							| 
									
										
										
										
											2012-04-25 07:23:20 +00:00
										 |  |  | 							sp->invalid = TRUE; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						mverts[*v].flag &= ~ME_VERT_TMP_TAG; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. */ | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 						PRINT_ERR("\tPoly %u needs missing edge (%u, %u)\n", sp->index, v1, v2); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 						if (do_fixes) | 
					
						
							|  |  |  | 							do_edge_recalc = TRUE; | 
					
						
							|  |  |  | 						else | 
					
						
							|  |  |  | 							sp->invalid = TRUE; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					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)); | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 							PRINT_ERR("\tLoop %u has invalid edge reference (%u), fixed using edge %u\n", | 
					
						
							|  |  |  | 							          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); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 							sp->invalid = TRUE; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											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)); | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 								PRINT_ERR("\tPoly %u has invalid edge reference (%u), fixed using edge %u\n", | 
					
						
							|  |  |  | 								          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); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 								sp->invalid = TRUE; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2011-02-09 15:13:20 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2011-02-09 01:27:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 				/* Now check that that poly does not use a same vertex more than once! */ | 
					
						
							|  |  |  | 				if (!sp->invalid) { | 
					
						
							|  |  |  | 					int *prev_v = v = sp->verts; | 
					
						
							|  |  |  | 					j = sp->numverts; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					qsort(sp->verts, j, sizeof(int), int_cmp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					for (j--, v++; j; j--, v++) { | 
					
						
							|  |  |  | 						if (*v != *prev_v) { | 
					
						
							|  |  |  | 							int dlt = v - prev_v; | 
					
						
							|  |  |  | 							if (dlt > 1) { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 								PRINT_ERR("\tPoly %u is invalid, it multi-uses vertex %u (%u times)\n", | 
					
						
							|  |  |  | 								          sp->index, *prev_v, dlt); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 								sp->invalid = TRUE; | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							prev_v = v; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-04-21 14:14:58 +00:00
										 |  |  | 					if (v - prev_v > 1) { /* Don't forget final verts! */ | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 						PRINT_ERR("\tPoly %u is invalid, it multi-uses vertex %u (%u times)\n", | 
					
						
							|  |  |  | 						          sp->index, *prev_v, (int)(v - prev_v)); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 						sp->invalid = TRUE; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 			int *p1_v = sp->verts, *p2_v = prev_sp->verts; | 
					
						
							|  |  |  | 			short p1_sub = TRUE, p2_sub = TRUE; | 
					
						
							|  |  |  | 			if (sp->invalid) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			/* Test same polys. */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 			/* 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++; | 
					
						
							| 
									
										
										
										
											2011-02-09 15:13:20 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 					/* Equality, both next verts. */ | 
					
						
							|  |  |  | 					p1_nv--; | 
					
						
							|  |  |  | 					p2_nv--; | 
					
						
							|  |  |  | 					p1_v++; | 
					
						
							|  |  |  | 					p2_v++; | 
					
						
							| 
									
										
										
										
											2011-02-09 15:13:20 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if (p1_nv && p1_sub) | 
					
						
							|  |  |  | 				p1_sub = FALSE; | 
					
						
							|  |  |  | 			else if (p2_nv && p2_sub) | 
					
						
							|  |  |  | 				p2_sub = FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (p1_sub && p2_sub) { | 
					
						
							| 
									
										
										
										
											2012-07-05 13:02:42 +00:00
										 |  |  | 				PRINT("\tPolys %u and %u use same vertices, considering poly %u as invalid.\n", | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 				      prev_sp->index, sp->index, sp->index); | 
					
						
							|  |  |  | 				sp->invalid = TRUE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			/* XXX In fact, these might be valid? :/ */ | 
					
						
							|  |  |  | 			else if (p1_sub) { | 
					
						
							| 
									
										
										
										
											2012-07-05 13:02:42 +00:00
										 |  |  | 				PRINT("\t%u is a sub-poly of %u, considering it as invalid.\n", sp->index, prev_sp->index); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 				sp->invalid = TRUE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if (p2_sub) { | 
					
						
							| 
									
										
										
										
											2012-07-05 13:02:42 +00:00
										 |  |  | 				PRINT("\t%u is a sub-poly of %u, considering it as invalid.\n", prev_sp->index, sp->index); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 				prev_sp->invalid = TRUE; | 
					
						
							|  |  |  | 				prev_sp = sp; /* sp is new reference poly. */ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 			if (0) { | 
					
						
							|  |  |  | 				p1_sub += 0; | 
					
						
							|  |  |  | 				p2_sub += 0; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 					PRINT_ERR("\tPolys %u and %u use same vertices (%u", | 
					
						
							|  |  |  | 					          prev_sp->index, sp->index, *p1_v); | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 					for (j = 1; j < p1_nv; j++) | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 						PRINT_ERR(", %u", p1_v[j]); | 
					
						
							|  |  |  | 					PRINT_ERR("), considering poly %u as invalid.\n", sp->index); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					is_valid = false; | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				sp->invalid = TRUE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | #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) { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 					PRINT_ERR("\tPolys %u and %u share loops from %u to %u, considering poly %u as invalid.\n", | 
					
						
							|  |  |  | 					          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++) { | 
					
						
							| 
									
										
										
										
											2011-12-08 04:51:03 +00:00
										 |  |  | 				/* note, greater then max defgroups is accounted for in our code, but not < 0 */ | 
					
						
							|  |  |  | 				if (!finite(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; | 
					
						
							|  |  |  | 						vert_weights_fixed = 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); | 
					
						
							| 
									
										
										
										
											2012-04-18 09:16:30 +00:00
										 |  |  | 						vert_weights_fixed = 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); | 
					
						
							|  |  |  | 						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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-18 09:16:30 +00:00
										 |  |  | 							vert_weights_fixed = TRUE; | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							| 
									
										
										
										
											2012-06-28 09:08:11 +00:00
										 |  |  | 		if (do_face_free) { | 
					
						
							|  |  |  | 			BKE_mesh_strip_loose_faces(mesh); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (do_polyloop_free) { | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 			BKE_mesh_strip_loose_polysloops(mesh); | 
					
						
							| 
									
										
										
										
											2011-02-09 15:13:20 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (do_edge_free) { | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 			BKE_mesh_strip_loose_edges(mesh); | 
					
						
							| 
									
										
										
										
											2011-02-09 15:13:20 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (do_edge_recalc) { | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 		int free_msel = FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		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) { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 				PRINT_ERR("\tMesh select element %d type %d index is negative, " | 
					
						
							|  |  |  | 				          "resetting selection stack.\n", i, msel->type); | 
					
						
							| 
									
										
										
										
											2012-07-02 09:57:31 +00:00
										 |  |  | 				free_msel = TRUE; | 
					
						
							|  |  |  | 				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) { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 				PRINT_ERR("\tMesh select element %d type %d index %d is larger than data array size %d, " | 
					
						
							|  |  |  | 				          "resetting selection stack.\n", i, msel->type, msel->index, tot_elem); | 
					
						
							| 
									
										
										
										
											2012-07-02 09:57:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				free_msel = TRUE; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (free_msel) { | 
					
						
							|  |  |  | 			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__); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	*r_change = (verts_fixed || vert_weights_fixed || do_polyloop_free || do_edge_free || do_edge_recalc || msel_fixed); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef PRINT
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-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-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. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | int BKE_mesh_validate(Mesh *me, const int do_verbose) | 
					
						
							| 
									
										
										
										
											2011-02-09 01:27:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 	bool is_valid = true; | 
					
						
							|  |  |  | 	bool is_change; | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							| 
									
										
										
										
											2013-09-04 06:50:15 +00:00
										 |  |  | 	        true, | 
					
						
							| 
									
										
										
										
											2013-09-04 01:29:34 +00:00
										 |  |  | 	        do_verbose, true, | 
					
						
							|  |  |  | 	        &is_change); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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, | 
					
						
							|  |  |  | 	        &is_change); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (is_change) { | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 		DAG_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-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++) { | 
					
						
							|  |  |  | 		int invalid = FALSE; | 
					
						
							|  |  |  | 		int i = p->loopstart; | 
					
						
							|  |  |  | 		int stop = i + p->totloop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (stop > me->totloop || stop < i) { | 
					
						
							|  |  |  | 			invalid = TRUE; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		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) { | 
					
						
							|  |  |  | 					invalid = TRUE; | 
					
						
							|  |  |  | 					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]; | 
					
						
							|  |  |  | 		int j, l_prev = (l + (mp->totloop - 1))->v; | 
					
						
							|  |  |  | 		for (j = 0; j < mp->totloop; j++, l++) { | 
					
						
							| 
									
										
										
										
											2012-03-15 20:10:07 +00:00
										 |  |  | 			if (!BLI_edgehash_haskey(eh, l_prev, l->v)) { | 
					
						
							|  |  |  | 				BLI_edgehash_insert(eh, l_prev, l->v, NULL); | 
					
						
							| 
									
										
										
										
											2011-09-01 08:27:35 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-04-18 09:16:30 +00:00
										 |  |  | 			l_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; | 
					
						
							|  |  |  | 	     BLI_edgehashIterator_isDone(ehi) == FALSE; | 
					
						
							|  |  |  | 	     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
										 |  |  | /** \} */ |