Cleanup: moar ugly G.main removal...
[blender.git] / source / blender / editors / physics / physics_pointcache.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) 2007 by Janne Karhu.
19  * All rights reserved.
20  *
21  * The Original Code is: all of this file.
22  *
23  * Contributor(s): none yet.
24  *
25  * ***** END GPL LICENSE BLOCK *****
26  */
27
28 /** \file blender/editors/physics/physics_pointcache.c
29  *  \ingroup edphys
30  */
31
32 #include <stdlib.h>
33 #include <string.h>
34
35 #include "MEM_guardedalloc.h"
36
37 #include "BLI_blenlib.h"
38 #include "BLI_utildefines.h"
39
40 #include "DNA_scene_types.h"
41
42 #include "BKE_context.h"
43 #include "BKE_screen.h"
44 #include "BKE_global.h"
45 #include "BKE_main.h"
46 #include "BKE_particle.h"
47 #include "BKE_pointcache.h"
48
49 #include "ED_particle.h"
50
51 #include "WM_api.h"
52 #include "WM_types.h"
53
54 #include "RNA_access.h"
55 #include "RNA_define.h"
56
57 #include "physics_intern.h"
58
59 static int ptcache_bake_all_poll(bContext *C)
60 {
61         return CTX_data_scene(C) != NULL;
62 }
63
64 static int ptcache_poll(bContext *C)
65 {
66         PointerRNA ptr= CTX_data_pointer_get_type(C, "point_cache", &RNA_PointCache);
67         return (ptr.data && ptr.id.data);
68 }
69
70 typedef struct PointCacheJob {
71         void *owner;
72         short *stop, *do_update;
73         float *progress;
74
75         PTCacheBaker *baker;
76 } PointCacheJob;
77
78 static void ptcache_job_free(void *customdata)
79 {
80         PointCacheJob *job = customdata;
81         MEM_freeN(job->baker);
82         MEM_freeN(job);
83 }
84
85 static int ptcache_job_break(void *customdata)
86 {
87         PointCacheJob *job = customdata;
88
89         if (G.is_break) {
90                 return 1;
91         }
92
93         if (job->stop && *(job->stop)) {
94                 return 1;
95         }
96
97         return 0;
98 }
99
100 static void ptcache_job_update(void *customdata, float progress, int *cancel)
101 {
102         PointCacheJob *job = customdata;
103
104         if (ptcache_job_break(job)) {
105                 *cancel = 1;
106         }
107
108         *(job->do_update) = true;
109         *(job->progress) = progress;
110 }
111
112 static void ptcache_job_startjob(void *customdata, short *stop, short *do_update, float *progress)
113 {
114         PointCacheJob *job = customdata;
115
116         job->stop = stop;
117         job->do_update = do_update;
118         job->progress = progress;
119
120         G.is_break = false;
121
122         /* XXX annoying hack: needed to prevent data corruption when changing
123          * scene frame in separate threads
124          */
125         G.is_rendering = true;
126         BKE_spacedata_draw_locks(true);
127
128         BKE_ptcache_bake(job->baker);
129
130         *do_update = true;
131         *stop = 0;
132 }
133
134 static void ptcache_job_endjob(void *customdata)
135 {
136         PointCacheJob *job = customdata;
137         Scene *scene = job->baker->scene;
138
139         G.is_rendering = false;
140         BKE_spacedata_draw_locks(false);
141
142         WM_set_locked_interface(G.main->wm.first, false);
143
144         WM_main_add_notifier(NC_SCENE | ND_FRAME, scene);
145         WM_main_add_notifier(NC_OBJECT | ND_POINTCACHE, job->baker->pid.ob);
146 }
147
148 static void ptcache_free_bake(PointCache *cache)
149 {
150         if (cache->edit) {
151                 if (!cache->edit->edited || 1) {// XXX okee("Lose changes done in particle mode?")) {
152                         PE_free_ptcache_edit(cache->edit);
153                         cache->edit = NULL;
154                         cache->flag &= ~PTCACHE_BAKED;
155                 }
156         }
157         else {
158                 cache->flag &= ~PTCACHE_BAKED;
159         }
160 }
161
162 static PTCacheBaker *ptcache_baker_create(bContext *C, wmOperator *op, bool all)
163 {
164         PTCacheBaker *baker = MEM_callocN(sizeof(PTCacheBaker), "PTCacheBaker");
165
166         baker->bmain = CTX_data_main(C);
167         baker->scene = CTX_data_scene(C);
168         baker->bake = RNA_boolean_get(op->ptr, "bake");
169         baker->render = 0;
170         baker->anim_init = 0;
171         baker->quick_step = 1;
172
173         if (!all) {
174                 PointerRNA ptr = CTX_data_pointer_get_type(C, "point_cache", &RNA_PointCache);
175                 Object *ob = ptr.id.data;
176                 PointCache *cache = ptr.data;
177
178                 ListBase pidlist;
179                 BKE_ptcache_ids_from_object(baker->bmain, &pidlist, ob, baker->scene, MAX_DUPLI_RECUR);
180
181                 for (PTCacheID *pid = pidlist.first; pid; pid = pid->next) {
182                         if (pid->cache == cache) {
183                                 baker->pid = *pid;
184                                 break;
185                         }
186                 }
187
188                 BLI_freelistN(&pidlist);
189         }
190
191         return baker;
192 }
193
194 static int ptcache_bake_exec(bContext *C, wmOperator *op)
195 {
196         bool all = STREQ(op->type->idname, "PTCACHE_OT_bake_all");
197
198         PTCacheBaker *baker = ptcache_baker_create(C, op, all);
199         BKE_ptcache_bake(baker);
200         MEM_freeN(baker);
201
202         return OPERATOR_FINISHED;
203 }
204
205 static int ptcache_bake_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event))
206 {
207         bool all = STREQ(op->type->idname, "PTCACHE_OT_bake_all");
208
209         PointCacheJob *job = MEM_mallocN(sizeof(PointCacheJob), "PointCacheJob");
210         job->baker = ptcache_baker_create(C, op, all);
211         job->baker->bake_job = job;
212         job->baker->update_progress = ptcache_job_update;
213
214         wmJob *wm_job = WM_jobs_get(CTX_wm_manager(C), CTX_wm_window(C), CTX_data_scene(C),
215                                     "Point Cache", WM_JOB_PROGRESS, WM_JOB_TYPE_POINTCACHE);
216
217         WM_jobs_customdata_set(wm_job, job, ptcache_job_free);
218         WM_jobs_timer(wm_job, 0.1, NC_OBJECT | ND_POINTCACHE, NC_OBJECT | ND_POINTCACHE);
219         WM_jobs_callbacks(wm_job, ptcache_job_startjob, NULL, NULL, ptcache_job_endjob);
220
221         WM_set_locked_interface(CTX_wm_manager(C), true);
222
223         WM_jobs_start(CTX_wm_manager(C), wm_job);
224
225         WM_event_add_modal_handler(C, op);
226
227         /* we must run modal until the bake job is done, otherwise the undo push
228          * happens before the job ends, which can lead to race conditions between
229          * the baking and file writing code */
230         return OPERATOR_RUNNING_MODAL;
231 }
232
233 static int ptcache_bake_modal(bContext *C, wmOperator *op, const wmEvent *UNUSED(event))
234 {
235         Scene *scene = (Scene *) op->customdata;
236
237         /* no running blender, remove handler and pass through */
238         if (0 == WM_jobs_test(CTX_wm_manager(C), scene, WM_JOB_TYPE_POINTCACHE)) {
239                 return OPERATOR_FINISHED | OPERATOR_PASS_THROUGH;
240         }
241
242         return OPERATOR_PASS_THROUGH;
243 }
244
245 static void ptcache_bake_cancel(bContext *C, wmOperator *op)
246 {
247         wmWindowManager *wm = CTX_wm_manager(C);
248         Scene *scene = (Scene *) op->customdata;
249
250         /* kill on cancel, because job is using op->reports */
251         WM_jobs_kill_type(wm, scene, WM_JOB_TYPE_POINTCACHE);
252 }
253
254 static int ptcache_free_bake_all_exec(bContext *C, wmOperator *UNUSED(op))
255 {
256         Main *bmain = CTX_data_main(C);
257         Scene *scene = CTX_data_scene(C);
258         Base *base;
259         PTCacheID *pid;
260         ListBase pidlist;
261
262         for (base = scene->base.first; base; base = base->next) {
263                 BKE_ptcache_ids_from_object(bmain, &pidlist, base->object, scene, MAX_DUPLI_RECUR);
264
265                 for (pid=pidlist.first; pid; pid=pid->next) {
266                         ptcache_free_bake(pid->cache);
267                 }
268
269                 BLI_freelistN(&pidlist);
270
271                 WM_event_add_notifier(C, NC_OBJECT|ND_POINTCACHE, base->object);
272         }
273
274         WM_event_add_notifier(C, NC_SCENE|ND_FRAME, scene);
275
276         return OPERATOR_FINISHED;
277 }
278
279 void PTCACHE_OT_bake_all(wmOperatorType *ot)
280 {
281         /* identifiers */
282         ot->name = "Bake All Physics";
283         ot->description = "Bake all physics";
284         ot->idname = "PTCACHE_OT_bake_all";
285
286         /* api callbacks */
287         ot->exec = ptcache_bake_exec;
288         ot->invoke = ptcache_bake_invoke;
289         ot->modal = ptcache_bake_modal;
290         ot->cancel = ptcache_bake_cancel;
291         ot->poll = ptcache_bake_all_poll;
292
293         /* flags */
294         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
295
296         RNA_def_boolean(ot->srna, "bake", 1, "Bake", "");
297 }
298 void PTCACHE_OT_free_bake_all(wmOperatorType *ot)
299 {
300         /* identifiers */
301         ot->name = "Free All Physics Bakes";
302         ot->idname = "PTCACHE_OT_free_bake_all";
303         ot->description = "Free all baked caches of all objects in the current scene";
304
305         /* api callbacks */
306         ot->exec = ptcache_free_bake_all_exec;
307         ot->poll = ptcache_bake_all_poll;
308
309         /* flags */
310         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
311 }
312
313 static int ptcache_free_bake_exec(bContext *C, wmOperator *UNUSED(op))
314 {
315         PointerRNA ptr= CTX_data_pointer_get_type(C, "point_cache", &RNA_PointCache);
316         PointCache *cache= ptr.data;
317         Object *ob= ptr.id.data;
318
319         ptcache_free_bake(cache);
320
321         WM_event_add_notifier(C, NC_OBJECT|ND_POINTCACHE, ob);
322
323         return OPERATOR_FINISHED;
324 }
325 static int ptcache_bake_from_cache_exec(bContext *C, wmOperator *UNUSED(op))
326 {
327         PointerRNA ptr= CTX_data_pointer_get_type(C, "point_cache", &RNA_PointCache);
328         PointCache *cache= ptr.data;
329         Object *ob= ptr.id.data;
330
331         cache->flag |= PTCACHE_BAKED;
332
333         WM_event_add_notifier(C, NC_OBJECT|ND_POINTCACHE, ob);
334
335         return OPERATOR_FINISHED;
336 }
337 void PTCACHE_OT_bake(wmOperatorType *ot)
338 {
339         /* identifiers */
340         ot->name = "Bake Physics";
341         ot->description = "Bake physics";
342         ot->idname = "PTCACHE_OT_bake";
343
344         /* api callbacks */
345         ot->exec = ptcache_bake_exec;
346         ot->invoke = ptcache_bake_invoke;
347         ot->modal = ptcache_bake_modal;
348         ot->cancel = ptcache_bake_cancel;
349         ot->poll = ptcache_poll;
350
351         /* flags */
352         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
353
354         RNA_def_boolean(ot->srna, "bake", 0, "Bake", "");
355 }
356 void PTCACHE_OT_free_bake(wmOperatorType *ot)
357 {
358         /* identifiers */
359         ot->name = "Free Physics Bake";
360         ot->description = "Free physics bake";
361         ot->idname = "PTCACHE_OT_free_bake";
362
363         /* api callbacks */
364         ot->exec = ptcache_free_bake_exec;
365         ot->poll = ptcache_poll;
366
367         /* flags */
368         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
369 }
370 void PTCACHE_OT_bake_from_cache(wmOperatorType *ot)
371 {
372         /* identifiers */
373         ot->name = "Bake From Cache";
374         ot->description = "Bake from cache";
375         ot->idname = "PTCACHE_OT_bake_from_cache";
376
377         /* api callbacks */
378         ot->exec = ptcache_bake_from_cache_exec;
379         ot->poll = ptcache_poll;
380
381         /* flags */
382         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
383 }
384
385 static int ptcache_add_new_exec(bContext *C, wmOperator *UNUSED(op))
386 {
387         Main *bmain = CTX_data_main(C);
388         Scene *scene = CTX_data_scene(C);
389         PointerRNA ptr= CTX_data_pointer_get_type(C, "point_cache", &RNA_PointCache);
390         Object *ob= ptr.id.data;
391         PointCache *cache= ptr.data;
392         PTCacheID *pid;
393         ListBase pidlist;
394
395         BKE_ptcache_ids_from_object(bmain, &pidlist, ob, scene, MAX_DUPLI_RECUR);
396
397         for (pid=pidlist.first; pid; pid=pid->next) {
398                 if (pid->cache == cache) {
399                         PointCache *cache_new = BKE_ptcache_add(pid->ptcaches);
400                         cache_new->step = pid->default_step;
401                         *(pid->cache_ptr) = cache_new;
402                         break;
403                 }
404         }
405
406         BLI_freelistN(&pidlist);
407
408         WM_event_add_notifier(C, NC_SCENE|ND_FRAME, scene);
409         WM_event_add_notifier(C, NC_OBJECT|ND_POINTCACHE, ob);
410
411         return OPERATOR_FINISHED;
412 }
413 static int ptcache_remove_exec(bContext *C, wmOperator *UNUSED(op))
414 {
415         Main *bmain = CTX_data_main(C);
416         PointerRNA ptr= CTX_data_pointer_get_type(C, "point_cache", &RNA_PointCache);
417         Scene *scene= CTX_data_scene(C);
418         Object *ob= ptr.id.data;
419         PointCache *cache= ptr.data;
420         PTCacheID *pid;
421         ListBase pidlist;
422
423         BKE_ptcache_ids_from_object(bmain, &pidlist, ob, scene, MAX_DUPLI_RECUR);
424
425         for (pid=pidlist.first; pid; pid=pid->next) {
426                 if (pid->cache == cache) {
427                         if (pid->ptcaches->first == pid->ptcaches->last)
428                                 continue; /* don't delete last cache */
429
430                         BLI_remlink(pid->ptcaches, pid->cache);
431                         BKE_ptcache_free(pid->cache);
432                         *(pid->cache_ptr) = pid->ptcaches->first;
433
434                         break;
435                 }
436         }
437
438         BLI_freelistN(&pidlist);
439
440         WM_event_add_notifier(C, NC_OBJECT|ND_POINTCACHE, ob);
441
442         return OPERATOR_FINISHED;
443 }
444 void PTCACHE_OT_add(wmOperatorType *ot)
445 {
446         /* identifiers */
447         ot->name = "Add New Cache";
448         ot->description = "Add new cache";
449         ot->idname = "PTCACHE_OT_add";
450
451         /* api callbacks */
452         ot->exec = ptcache_add_new_exec;
453         ot->poll = ptcache_poll;
454
455         /* flags */
456         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
457 }
458 void PTCACHE_OT_remove(wmOperatorType *ot)
459 {
460         /* identifiers */
461         ot->name = "Delete Current Cache";
462         ot->description = "Delete current cache";
463         ot->idname = "PTCACHE_OT_remove";
464
465         /* api callbacks */
466         ot->exec = ptcache_remove_exec;
467         ot->poll = ptcache_poll;
468
469         /* flags */
470         ot->flag = OPTYPE_REGISTER|OPTYPE_UNDO;
471 }
472