Fix smoke to render in Cycles again.
[blender.git] / source / blender / modifiers / intern / MOD_smoke.c
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software  Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  * The Original Code is Copyright (C) 2005 by the Blender Foundation.
19  * All rights reserved.
20  *
21  * Contributor(s): Daniel Dunbar
22  *                 Ton Roosendaal,
23  *                 Ben Batt,
24  *                 Brecht Van Lommel,
25  *                 Campbell Barton
26  *
27  * ***** END GPL LICENSE BLOCK *****
28  *
29  */
30
31 /** \file blender/modifiers/intern/MOD_smoke.c
32  *  \ingroup modifiers
33  */
34
35
36 #include <stddef.h>
37
38 #include "MEM_guardedalloc.h"
39
40 #include "DNA_collection_types.h"
41 #include "DNA_object_types.h"
42 #include "DNA_scene_types.h"
43 #include "DNA_smoke_types.h"
44 #include "DNA_object_force_types.h"
45 #include "DNA_mesh_types.h"
46
47 #include "BLI_utildefines.h"
48
49 #include "BKE_cdderivedmesh.h"
50 #include "BKE_layer.h"
51 #include "BKE_library.h"
52 #include "BKE_library_query.h"
53 #include "BKE_main.h"
54 #include "BKE_modifier.h"
55 #include "BKE_smoke.h"
56
57 #include "DEG_depsgraph.h"
58 #include "DEG_depsgraph_build.h"
59 #include "DEG_depsgraph_physics.h"
60 #include "DEG_depsgraph_query.h"
61
62 #include "MOD_modifiertypes.h"
63
64 static void initData(ModifierData *md)
65 {
66         SmokeModifierData *smd = (SmokeModifierData *) md;
67
68         smd->domain = NULL;
69         smd->flow = NULL;
70         smd->coll = NULL;
71         smd->type = 0;
72         smd->time = -1;
73 }
74
75 static void copyData(const ModifierData *md, ModifierData *target, const int flag)
76 {
77         const SmokeModifierData *smd  = (const SmokeModifierData *)md;
78         SmokeModifierData *tsmd = (SmokeModifierData *)target;
79
80         smokeModifier_free(tsmd);
81         smokeModifier_copy(smd, tsmd, flag);
82 }
83
84 static void freeData(ModifierData *md)
85 {
86         SmokeModifierData *smd = (SmokeModifierData *) md;
87
88         smokeModifier_free(smd);
89 }
90
91 static CustomDataMask requiredDataMask(Object *UNUSED(ob), ModifierData *md)
92 {
93         SmokeModifierData *smd  = (SmokeModifierData *)md;
94         CustomDataMask dataMask = 0;
95
96         if (smd && (smd->type & MOD_SMOKE_TYPE_FLOW) && smd->flow) {
97                 if (smd->flow->source == MOD_SMOKE_FLOW_SOURCE_MESH) {
98                         /* vertex groups */
99                         if (smd->flow->vgroup_density)
100                                 dataMask |= CD_MASK_MDEFORMVERT;
101                         /* uv layer */
102                         if (smd->flow->texture_type == MOD_SMOKE_FLOW_TEXTURE_MAP_UV)
103                                 dataMask |= CD_MASK_MTFACE;
104                 }
105         }
106         return dataMask;
107 }
108
109 static Mesh *applyModifier(
110         ModifierData *md, const ModifierEvalContext *ctx,
111         Mesh *me)
112 {
113         SmokeModifierData *smd = (SmokeModifierData *) md;
114
115         if (ctx->flag & MOD_APPLY_ORCO) {
116                 return me;
117         }
118
119         Scene *scene = DEG_get_evaluated_scene(ctx->depsgraph);
120         return smokeModifier_do(smd, ctx->depsgraph, scene, ctx->object, me);
121 }
122
123 static bool dependsOnTime(ModifierData *UNUSED(md))
124 {
125         return true;
126 }
127
128 static bool is_flow_cb(Object *UNUSED(ob), ModifierData *md)
129 {
130         SmokeModifierData *smd = (SmokeModifierData *) md;
131         return (smd->type & MOD_SMOKE_TYPE_FLOW) && smd->flow;
132 }
133
134 static bool is_coll_cb(Object *UNUSED(ob), ModifierData *md)
135 {
136         SmokeModifierData *smd = (SmokeModifierData *) md;
137         return (smd->type & MOD_SMOKE_TYPE_COLL) && smd->coll;
138 }
139
140 static void updateDepsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
141 {
142         SmokeModifierData *smd = (SmokeModifierData *)md;
143
144         if (smd && (smd->type & MOD_SMOKE_TYPE_DOMAIN) && smd->domain) {
145                 DEG_add_collision_relations(ctx->node, ctx->object, smd->domain->fluid_group, eModifierType_Smoke, is_flow_cb, "Smoke Flow");
146                 DEG_add_collision_relations(ctx->node, ctx->object, smd->domain->coll_group, eModifierType_Smoke, is_coll_cb, "Smoke Coll");
147                 DEG_add_forcefield_relations(ctx->node, ctx->object, smd->domain->effector_weights, true, PFIELD_SMOKEFLOW, "Smoke Force Field");
148         }
149 }
150
151 static void foreachIDLink(
152         ModifierData *md, Object *ob,
153         IDWalkFunc walk, void *userData)
154 {
155         SmokeModifierData *smd = (SmokeModifierData *) md;
156
157         if (smd->type == MOD_SMOKE_TYPE_DOMAIN && smd->domain) {
158                 walk(userData, ob, (ID **)&smd->domain->coll_group, IDWALK_CB_NOP);
159                 walk(userData, ob, (ID **)&smd->domain->fluid_group, IDWALK_CB_NOP);
160                 walk(userData, ob, (ID **)&smd->domain->eff_group, IDWALK_CB_NOP);
161
162                 if (smd->domain->effector_weights) {
163                         walk(userData, ob, (ID **)&smd->domain->effector_weights->group, IDWALK_CB_NOP);
164                 }
165         }
166
167         if (smd->type == MOD_SMOKE_TYPE_FLOW && smd->flow) {
168                 walk(userData, ob, (ID **)&smd->flow->noise_texture, IDWALK_CB_USER);
169         }
170 }
171
172 ModifierTypeInfo modifierType_Smoke = {
173         /* name */              "Smoke",
174         /* structName */        "SmokeModifierData",
175         /* structSize */        sizeof(SmokeModifierData),
176         /* type */              eModifierTypeType_Constructive,
177         /* flags */             eModifierTypeFlag_AcceptsMesh |
178                                 eModifierTypeFlag_UsesPointCache |
179                                 eModifierTypeFlag_Single,
180
181         /* copyData */          copyData,
182
183         /* deformVerts_DM */    NULL,
184         /* deformMatrices_DM */ NULL,
185         /* deformVertsEM_DM */  NULL,
186         /* deformMatricesEM_DM*/NULL,
187         /* applyModifier_DM */  NULL,
188         /* applyModifierEM_DM */NULL,
189
190         /* deformVerts */       NULL,
191         /* deformMatrices */    NULL,
192         /* deformVertsEM */     NULL,
193         /* deformMatricesEM */  NULL,
194         /* applyModifier */     applyModifier,
195         /* applyModifierEM */   NULL,
196
197         /* initData */          initData,
198         /* requiredDataMask */  requiredDataMask,
199         /* freeData */          freeData,
200         /* isDisabled */        NULL,
201         /* updateDepsgraph */   updateDepsgraph,
202         /* dependsOnTime */     dependsOnTime,
203         /* dependsOnNormals */  NULL,
204         /* foreachObjectLink */ NULL,
205         /* foreachIDLink */     foreachIDLink,
206         /* foreachTexLink */    NULL
207 };