Welcome to mirror list, hosted at ThFree Co, Russian Federation.

git.blender.org/blender.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSergey Sharybin <sergey.vfx@gmail.com>2014-05-03 17:58:37 +0400
committerSergey Sharybin <sergey.vfx@gmail.com>2014-05-03 18:13:01 +0400
commite08db08a84bffaab27bc4562fd41f44756eb2e3e (patch)
tree56f9c70d7c28f4fd99f8f7e150035c9936b842a5 /source/blender/modifiers
parent48c1a7f8a45f1474c1484950687f0f30612a1731 (diff)
Fix T39997: Multiple boolean modifiers sharing the same right operand crashes
The issue was caused by the temporary CD layers being allocated for subsurf meshes, same as we've got back in 881fb43. In the long run this temporary storage is to be re-considered, but it'll also imply re-considering of the Derivedmesh interaction as well. For now let's use a simpler solution which is forbidding modifiers to call getArray for other objects' derivedMeshes but use an API calls which would allocate local copy of the data preventing race condition of shared data in DM.
Diffstat (limited to 'source/blender/modifiers')
-rw-r--r--source/blender/modifiers/intern/MOD_boolean_util.c88
1 files changed, 69 insertions, 19 deletions
diff --git a/source/blender/modifiers/intern/MOD_boolean_util.c b/source/blender/modifiers/intern/MOD_boolean_util.c
index 251c5a8e311..99016a0a41b 100644
--- a/source/blender/modifiers/intern/MOD_boolean_util.c
+++ b/source/blender/modifiers/intern/MOD_boolean_util.c
@@ -27,11 +27,12 @@
* \ingroup modifiers
*/
+#include "MEM_guardedalloc.h"
+
#include "DNA_material_types.h"
#include "DNA_meshdata_types.h"
#include "DNA_object_types.h"
-
#include "BLI_utildefines.h"
#include "BLI_alloca.h"
#include "BLI_ghash.h"
@@ -88,6 +89,41 @@ static void DM_loop_interp_from_poly(DerivedMesh *source_dm,
source_poly->totloop, target_loop_index);
}
+typedef struct DMArrays {
+ MVert *mvert;
+ MEdge *medge;
+ MLoop *mloop;
+ MPoly *mpoly;
+ bool mvert_allocated;
+ bool medge_allocated;
+ bool mloop_allocated;
+ bool mpoly_allocated;
+} DMArrays;
+
+static void dm_arrays_get(DerivedMesh *dm, DMArrays *arrays)
+{
+ arrays->mvert = DM_get_vert_array(dm, &arrays->mvert_allocated);
+ arrays->medge = DM_get_edge_array(dm, &arrays->medge_allocated);
+ arrays->mloop = DM_get_loop_array(dm, &arrays->mloop_allocated);
+ arrays->mpoly = DM_get_poly_array(dm, &arrays->mpoly_allocated);
+}
+
+static void dm_arrays_free(DMArrays *arrays)
+{
+ if (arrays->mvert_allocated) {
+ MEM_freeN(arrays->mvert);
+ }
+ if (arrays->medge_allocated) {
+ MEM_freeN(arrays->medge);
+ }
+ if (arrays->mloop_allocated) {
+ MEM_freeN(arrays->mloop);
+ }
+ if (arrays->mpoly_allocated) {
+ MEM_freeN(arrays->mpoly);
+ }
+}
+
/* **** Importer from derived mesh to Carve **** */
typedef struct ImportMeshData {
@@ -633,25 +669,30 @@ static int operation_from_optype(int int_op_type)
return operation;
}
-static void prepare_import_data(Object *object, DerivedMesh *dm, ImportMeshData *import_data)
+static void prepare_import_data(Object *object,
+ DerivedMesh *dm,
+ const DMArrays *dm_arrays,
+ ImportMeshData *import_data)
{
import_data->dm = dm;
copy_m4_m4(import_data->obmat, object->obmat);
- import_data->mvert = dm->getVertArray(dm);
- import_data->medge = dm->getEdgeArray(dm);
- import_data->mloop = dm->getLoopArray(dm);
- import_data->mpoly = dm->getPolyArray(dm);
+ import_data->mvert = dm_arrays->mvert;
+ import_data->medge = dm_arrays->medge;
+ import_data->mloop = dm_arrays->mloop;
+ import_data->mpoly = dm_arrays->mpoly;
}
-static struct CarveMeshDescr *carve_mesh_from_dm(Object *object, DerivedMesh *dm)
+static struct CarveMeshDescr *carve_mesh_from_dm(Object *object,
+ DerivedMesh *dm,
+ const DMArrays *dm_arrays)
{
ImportMeshData import_data;
- prepare_import_data(object, dm, &import_data);
+ prepare_import_data(object, dm, dm_arrays, &import_data);
return carve_addMesh(&import_data, &MeshImporter);
}
-static void prepare_export_data(Object *object_left, DerivedMesh *dm_left,
- Object *object_right, DerivedMesh *dm_right,
+static void prepare_export_data(Object *object_left, DerivedMesh *dm_left, const DMArrays *dm_left_arrays,
+ Object *object_right, DerivedMesh *dm_right, const DMArrays *dm_right_arrays,
ExportMeshData *export_data)
{
float object_right_imat[4][4];
@@ -664,12 +705,12 @@ static void prepare_export_data(Object *object_left, DerivedMesh *dm_left,
export_data->dm_left = dm_left;
export_data->dm_right = dm_right;
- export_data->mvert_left = dm_left->getVertArray(dm_left);
- export_data->mloop_left = dm_left->getLoopArray(dm_left);
- export_data->mpoly_left = dm_left->getPolyArray(dm_left);
- export_data->mvert_right = dm_right->getVertArray(dm_right);
- export_data->mloop_right = dm_right->getLoopArray(dm_right);
- export_data->mpoly_right = dm_right->getPolyArray(dm_right);
+ export_data->mvert_left = dm_left_arrays->mvert;
+ export_data->mloop_left = dm_left_arrays->mloop;
+ export_data->mpoly_left = dm_left_arrays->mpoly;
+ export_data->mvert_right = dm_right_arrays->mvert;
+ export_data->mloop_right = dm_right_arrays->mloop;
+ export_data->mpoly_right = dm_right_arrays->mpoly;
export_data->material_hash = BLI_ghash_ptr_new("CSG_mat gh");
@@ -690,6 +731,7 @@ DerivedMesh *NewBooleanDerivedMesh(DerivedMesh *dm, struct Object *ob,
DerivedMesh *output_dm = NULL;
int operation;
bool result;
+ DMArrays dm_left_arrays, dm_right_arrays;
if (dm == NULL || dm_select == NULL) {
return NULL;
@@ -700,8 +742,11 @@ DerivedMesh *NewBooleanDerivedMesh(DerivedMesh *dm, struct Object *ob,
return NULL;
}
- left = carve_mesh_from_dm(ob_select, dm_select);
- right = carve_mesh_from_dm(ob, dm);
+ dm_arrays_get(dm_select, &dm_left_arrays);
+ dm_arrays_get(dm, &dm_right_arrays);
+
+ left = carve_mesh_from_dm(ob_select, dm_select, &dm_left_arrays);
+ right = carve_mesh_from_dm(ob, dm, &dm_right_arrays);
result = carve_performBooleanOperation(left, right, operation, &output);
@@ -711,7 +756,9 @@ DerivedMesh *NewBooleanDerivedMesh(DerivedMesh *dm, struct Object *ob,
if (result) {
ExportMeshData export_data;
- prepare_export_data(ob_select, dm_select, ob, dm, &export_data);
+ prepare_export_data(ob_select, dm_select, &dm_left_arrays,
+ ob, dm, &dm_right_arrays,
+ &export_data);
carve_exportMesh(output, &MeshExporter, &export_data);
output_dm = export_data.dm;
@@ -723,5 +770,8 @@ DerivedMesh *NewBooleanDerivedMesh(DerivedMesh *dm, struct Object *ob,
carve_deleteMesh(output);
}
+ dm_arrays_free(&dm_left_arrays);
+ dm_arrays_free(&dm_right_arrays);
+
return output_dm;
}