Recent spinlock commit made scheduling unsafe for threading
[blender.git] / source / blender / blenkernel / intern / blender.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) 2001-2002 by NaN Holding BV.
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/blenkernel/intern/blender.c
29  *  \ingroup bke
30  */
31
32
33 #ifndef _WIN32 
34 #  include <unistd.h> // for read close
35 #else
36 #  include <io.h> // for open close read
37 #  define open _open
38 #  define read _read
39 #  define close _close
40 #  define write _write
41 #endif
42
43 #include <stdlib.h>
44 #include <stdio.h>
45 #include <stddef.h>
46 #include <string.h>
47 #include <fcntl.h>  /* for open */
48 #include <errno.h>
49
50 #include "MEM_guardedalloc.h"
51
52 #include "DNA_userdef_types.h"
53 #include "DNA_scene_types.h"
54 #include "DNA_screen_types.h"
55 #include "DNA_sequence_types.h"
56 #include "DNA_sound_types.h"
57 #include "DNA_windowmanager_types.h"
58
59 #include "BLI_blenlib.h"
60 #include "BLI_dynstr.h"
61 #include "BLI_utildefines.h"
62 #include "BLI_callbacks.h"
63
64 #include "IMB_imbuf.h"
65 #include "IMB_moviecache.h"
66
67 #include "BKE_blender.h"
68 #include "BKE_bpath.h"
69 #include "BKE_brush.h"
70 #include "BKE_context.h"
71 #include "BKE_depsgraph.h"
72 #include "BKE_displist.h"
73 #include "BKE_global.h"
74 #include "BKE_idprop.h"
75 #include "BKE_image.h"
76 #include "BKE_ipo.h"
77 #include "BKE_library.h"
78 #include "BKE_main.h"
79 #include "BKE_node.h"
80 #include "BKE_report.h"
81 #include "BKE_scene.h"
82 #include "BKE_screen.h"
83 #include "BKE_sequencer.h"
84 #include "BKE_sound.h"
85
86 #include "RE_pipeline.h"
87
88 #include "BLF_api.h"
89
90 #include "BLO_undofile.h"
91 #include "BLO_readfile.h" 
92 #include "BLO_writefile.h" 
93
94 #include "RNA_access.h"
95
96 #include "WM_api.h" // XXXXX BAD, very BAD dependency (bad level call) - remove asap, elubie
97
98 #include "IMB_colormanagement.h"
99
100 #ifdef WITH_PYTHON
101 #  include "BPY_extern.h"
102 #endif
103
104 Global G;
105 UserDef U;
106 /* ListBase = {NULL, NULL}; */
107
108 char versionstr[48] = "";
109
110 /* ********** free ********** */
111
112 /* only to be called on exit blender */
113 void free_blender(void)
114 {
115         /* samples are in a global list..., also sets G.main->sound->sample NULL */
116         free_main(G.main);
117         G.main = NULL;
118
119         BKE_spacetypes_free();      /* after free main, it uses space callbacks */
120         
121         IMB_exit();
122         BKE_images_exit();
123         DAG_threaded_exit();
124
125         BKE_brush_system_exit();
126
127         BLI_callback_global_finalize();
128
129         BKE_sequencer_cache_destruct();
130         IMB_moviecache_destruct();
131         
132         free_nodesystem();
133 }
134
135 void initglobals(void)
136 {
137         memset(&G, 0, sizeof(Global));
138         
139         U.savetime = 1;
140
141         G.main = MEM_callocN(sizeof(Main), "initglobals");
142
143         strcpy(G.ima, "//");
144
145         if (BLENDER_SUBVERSION)
146                 BLI_snprintf(versionstr, sizeof(versionstr), "v%d.%02d.%d", BLENDER_VERSION / 100, BLENDER_VERSION % 100, BLENDER_SUBVERSION);
147         else
148                 BLI_snprintf(versionstr, sizeof(versionstr), "v%d.%02d", BLENDER_VERSION / 100, BLENDER_VERSION % 100);
149
150 #ifdef _WIN32
151         G.windowstate = 0;
152 #endif
153
154 #ifndef WITH_PYTHON_SECURITY /* default */
155         G.f |= G_SCRIPT_AUTOEXEC;
156 #else
157         G.f &= ~G_SCRIPT_AUTOEXEC;
158 #endif
159 }
160
161 /***/
162
163 static void clear_global(void) 
164 {
165 //      extern short winqueue_break;    /* screen.c */
166
167         free_main(G.main);          /* free all lib data */
168         
169 //      free_vertexpaint();
170
171         G.main = NULL;
172 }
173
174 static bool clean_paths_visit_cb(void *UNUSED(userdata), char *path_dst, const char *path_src)
175 {
176         strcpy(path_dst, path_src);
177         BLI_clean(path_dst);
178         return !STREQ(path_dst, path_src);
179 }
180
181 /* make sure path names are correct for OS */
182 static void clean_paths(Main *main)
183 {
184         Scene *scene;
185
186         BKE_bpath_traverse_main(main, clean_paths_visit_cb, BKE_BPATH_TRAVERSE_SKIP_MULTIFILE, NULL);
187
188         for (scene = main->scene.first; scene; scene = scene->id.next) {
189                 BLI_clean(scene->r.pic);
190         }
191 }
192
193 /* context matching */
194 /* handle no-ui case */
195
196 /* note, this is called on Undo so any slow conversion functions here
197  * should be avoided or check (mode!='u') */
198
199 static void setup_app_data(bContext *C, BlendFileData *bfd, const char *filepath)
200 {
201         bScreen *curscreen = NULL;
202         Scene *curscene = NULL;
203         int recover;
204         char mode;
205
206         /* 'u' = undo save, 'n' = no UI load */
207         if (bfd->main->screen.first == NULL) mode = 'u';
208         else if (G.fileflags & G_FILE_NO_UI) mode = 'n';
209         else mode = 0;
210
211         recover = (G.fileflags & G_FILE_RECOVER);
212
213         /* Free all render results, without this stale data gets displayed after loading files */
214         if (mode != 'u') {
215                 RE_FreeAllRenderResults();
216         }
217
218         /* Only make filepaths compatible when loading for real (not undo) */
219         if (mode != 'u') {
220                 clean_paths(bfd->main);
221         }
222
223         /* XXX here the complex windowmanager matching */
224         
225         /* no load screens? */
226         if (mode) {
227                 /* comes from readfile.c */
228                 SWAP(ListBase, G.main->wm, bfd->main->wm);
229                 SWAP(ListBase, G.main->screen, bfd->main->screen);
230                 SWAP(ListBase, G.main->script, bfd->main->script);
231                 
232                 /* we re-use current screen */
233                 curscreen = CTX_wm_screen(C);
234                 /* but use new Scene pointer */
235                 curscene = bfd->curscene;
236                 if (curscene == NULL) curscene = bfd->main->scene.first;
237                 /* empty file, we add a scene to make Blender work */
238                 if (curscene == NULL) curscene = BKE_scene_add(bfd->main, "Empty");
239                 
240                 /* and we enforce curscene to be in current screen */
241                 if (curscreen) curscreen->scene = curscene;  /* can run in bgmode */
242
243                 /* clear_global will free G.main, here we can still restore pointers */
244                 blo_lib_link_screen_restore(bfd->main, curscreen, curscene);
245         }
246         
247         /* free G.main Main database */
248 //      CTX_wm_manager_set(C, NULL);
249         clear_global();
250         
251         /* clear old property update cache, in case some old references are left dangling */
252         RNA_property_update_cache_free();
253         
254         G.main = bfd->main;
255
256         CTX_data_main_set(C, G.main);
257
258         sound_init_main(G.main);
259         
260         if (bfd->user) {
261                 
262                 /* only here free userdef themes... */
263                 BKE_userdef_free();
264                 
265                 U = *bfd->user;
266                 MEM_freeN(bfd->user);
267         }
268         
269         /* case G_FILE_NO_UI or no screens in file */
270         if (mode) {
271                 /* leave entire context further unaltered? */
272                 CTX_data_scene_set(C, curscene);
273         }
274         else {
275                 G.winpos = bfd->winpos;
276                 G.displaymode = bfd->displaymode;
277                 G.fileflags = bfd->fileflags;
278                 CTX_wm_manager_set(C, G.main->wm.first);
279                 CTX_wm_screen_set(C, bfd->curscreen);
280                 CTX_data_scene_set(C, bfd->curscene);
281                 CTX_wm_area_set(C, NULL);
282                 CTX_wm_region_set(C, NULL);
283                 CTX_wm_menu_set(C, NULL);
284         }
285         
286         /* this can happen when active scene was lib-linked, and doesn't exist anymore */
287         if (CTX_data_scene(C) == NULL) {
288                 /* in case we don't even have a local scene, add one */
289                 if (!G.main->scene.first)
290                         BKE_scene_add(G.main, "Scene");
291
292                 CTX_data_scene_set(C, G.main->scene.first);
293                 CTX_wm_screen(C)->scene = CTX_data_scene(C);
294                 curscene = CTX_data_scene(C);
295         }
296
297         /* special cases, override loaded flags: */
298         if (G.f != bfd->globalf) {
299                 const int flags_keep = (G_SWAP_EXCHANGE | G_SCRIPT_AUTOEXEC | G_SCRIPT_OVERRIDE_PREF);
300                 bfd->globalf = (bfd->globalf & ~flags_keep) | (G.f & flags_keep);
301         }
302
303
304         G.f = bfd->globalf;
305
306 #ifdef WITH_PYTHON
307         /* let python know about new main */
308         BPY_context_update(C);
309 #endif
310
311         if (!G.background) {
312                 //setscreen(G.curscreen);
313         }
314         
315         /* FIXME: this version patching should really be part of the file-reading code,
316          * but we still get too many unrelated data-corruption crashes otherwise... */
317         if (G.main->versionfile < 250)
318                 do_versions_ipos_to_animato(G.main);
319         
320         G.main->recovered = 0;
321         
322         /* startup.blend or recovered startup */
323         if (bfd->filename[0] == 0) {
324                 G.main->name[0] = 0;
325         }
326         else if (recover && G.relbase_valid) {
327                 /* in case of autosave or quit.blend, use original filename instead
328                  * use relbase_valid to make sure the file is saved, else we get <memory2> in the filename */
329                 filepath = bfd->filename;
330                 G.main->recovered = 1;
331         
332                 /* these are the same at times, should never copy to the same location */
333                 if (G.main->name != filepath)
334                         BLI_strncpy(G.main->name, filepath, FILE_MAX);
335         }
336         
337         /* baseflags, groups, make depsgraph, etc */
338         /* first handle case if other windows have different scenes visible */
339         if (mode == 0) {
340                 wmWindowManager *wm = G.main->wm.first;
341                 
342                 if (wm) {
343                         wmWindow *win;
344                         
345                         for (win = wm->windows.first; win; win = win->next) {
346                                 if (win->screen && win->screen->scene) /* zealous check... */
347                                         if (win->screen->scene != CTX_data_scene(C))
348                                                 BKE_scene_set_background(G.main, win->screen->scene);
349                         }
350                 }
351         }
352         BKE_scene_set_background(G.main, CTX_data_scene(C));
353
354         if (mode != 'u') {
355                 IMB_colormanagement_check_file_config(G.main);
356         }
357
358         MEM_freeN(bfd);
359
360 }
361
362 static int handle_subversion_warning(Main *main, ReportList *reports)
363 {
364         if (main->minversionfile > BLENDER_VERSION ||
365             (main->minversionfile == BLENDER_VERSION &&
366              main->minsubversionfile > BLENDER_SUBVERSION))
367         {
368                 BKE_reportf(reports, RPT_ERROR, "File written by newer Blender binary (%d.%d), expect loss of data!",
369                             main->minversionfile, main->minsubversionfile);
370         }
371
372         return 1;
373 }
374
375 static void keymap_item_free(wmKeyMapItem *kmi)
376 {
377         if (kmi->properties) {
378                 IDP_FreeProperty(kmi->properties);
379                 MEM_freeN(kmi->properties);
380         }
381         if (kmi->ptr)
382                 MEM_freeN(kmi->ptr);
383 }
384
385 void BKE_userdef_free(void)
386 {
387         wmKeyMap *km;
388         wmKeyMapItem *kmi;
389         wmKeyMapDiffItem *kmdi;
390         bAddon *addon, *addon_next;
391
392         for (km = U.user_keymaps.first; km; km = km->next) {
393                 for (kmdi = km->diff_items.first; kmdi; kmdi = kmdi->next) {
394                         if (kmdi->add_item) {
395                                 keymap_item_free(kmdi->add_item);
396                                 MEM_freeN(kmdi->add_item);
397                         }
398                         if (kmdi->remove_item) {
399                                 keymap_item_free(kmdi->remove_item);
400                                 MEM_freeN(kmdi->remove_item);
401                         }
402                 }
403
404                 for (kmi = km->items.first; kmi; kmi = kmi->next)
405                         keymap_item_free(kmi);
406
407                 BLI_freelistN(&km->diff_items);
408                 BLI_freelistN(&km->items);
409         }
410         
411         for (addon = U.addons.first; addon; addon = addon_next) {
412                 addon_next = addon->next;
413                 if (addon->prop) {
414                         IDP_FreeProperty(addon->prop);
415                         MEM_freeN(addon->prop);
416                 }
417                 MEM_freeN(addon);
418         }
419
420         BLI_freelistN(&U.autoexec_paths);
421
422         BLI_freelistN(&U.uistyles);
423         BLI_freelistN(&U.uifonts);
424         BLI_freelistN(&U.themes);
425         BLI_freelistN(&U.user_keymaps);
426 }
427
428 /* handle changes in settings that need recalc */
429 void BKE_userdef_state(void)
430 {
431         /* prevent accidents */
432         if (U.pixelsize == 0) U.pixelsize = 1;
433         
434         BLF_default_dpi(U.pixelsize * U.dpi);
435         U.widget_unit = (U.pixelsize * U.dpi * 20 + 36) / 72;
436
437 }
438
439 int BKE_read_file(bContext *C, const char *filepath, ReportList *reports)
440 {
441         BlendFileData *bfd;
442         int retval = BKE_READ_FILE_OK;
443
444         if (strstr(filepath, BLENDER_STARTUP_FILE) == NULL) /* don't print user-pref loading */
445                 printf("read blend: %s\n", filepath);
446
447         bfd = BLO_read_from_file(filepath, reports);
448         if (bfd) {
449                 if (bfd->user) retval = BKE_READ_FILE_OK_USERPREFS;
450                 
451                 if (0 == handle_subversion_warning(bfd->main, reports)) {
452                         free_main(bfd->main);
453                         MEM_freeN(bfd);
454                         bfd = NULL;
455                         retval = BKE_READ_FILE_FAIL;
456                 }
457                 else
458                         setup_app_data(C, bfd, filepath);  // frees BFD
459         }
460         else
461                 BKE_reports_prependf(reports, "Loading '%s' failed: ", filepath);
462                 
463         return (bfd ? retval : BKE_READ_FILE_FAIL);
464 }
465
466 int BKE_read_file_from_memory(bContext *C, const void *filebuf, int filelength, ReportList *reports)
467 {
468         BlendFileData *bfd;
469
470         bfd = BLO_read_from_memory(filebuf, filelength, reports);
471         if (bfd)
472                 setup_app_data(C, bfd, "<memory2>");
473         else
474                 BKE_reports_prepend(reports, "Loading failed: ");
475
476         return (bfd ? 1 : 0);
477 }
478
479 /* memfile is the undo buffer */
480 int BKE_read_file_from_memfile(bContext *C, MemFile *memfile, ReportList *reports)
481 {
482         BlendFileData *bfd;
483
484         bfd = BLO_read_from_memfile(CTX_data_main(C), G.main->name, memfile, reports);
485         if (bfd) {
486                 /* remove the unused screens and wm */
487                 while (bfd->main->wm.first)
488                         BKE_libblock_free(&bfd->main->wm, bfd->main->wm.first);
489                 while (bfd->main->screen.first)
490                         BKE_libblock_free(&bfd->main->screen, bfd->main->screen.first);
491                 
492                 setup_app_data(C, bfd, "<memory1>");
493         }
494         else
495                 BKE_reports_prepend(reports, "Loading failed: ");
496
497         return (bfd ? 1 : 0);
498 }
499
500 /* only read the userdef from a .blend */
501 int BKE_read_file_userdef(const char *filepath, ReportList *reports)
502 {
503         BlendFileData *bfd;
504         int retval = 0;
505         
506         bfd = BLO_read_from_file(filepath, reports);
507         if (bfd->user) {
508                 retval = BKE_READ_FILE_OK_USERPREFS;
509                 
510                 /* only here free userdef themes... */
511                 BKE_userdef_free();
512                 
513                 U = *bfd->user;
514                 MEM_freeN(bfd->user);
515         }
516         free_main(bfd->main);
517         MEM_freeN(bfd);
518         
519         return retval;
520 }
521
522 /* only write the userdef in a .blend */
523 int BKE_write_file_userdef(const char *filepath, ReportList *reports)
524 {
525         Main *mainb = MEM_callocN(sizeof(Main), "empty main");
526         int retval = 0;
527         
528         if (BLO_write_file(mainb, filepath, G_FILE_USERPREFS, reports, NULL)) {
529                 retval = 1;
530         }
531         
532         MEM_freeN(mainb);
533         
534         return retval;
535 }
536
537 /* *****************  testing for break ************* */
538
539 static void (*blender_test_break_cb)(void) = NULL;
540
541 void set_blender_test_break_cb(void (*func)(void) )
542 {
543         blender_test_break_cb = func;
544 }
545
546
547 int blender_test_break(void)
548 {
549         if (!G.background) {
550                 if (blender_test_break_cb)
551                         blender_test_break_cb();
552         }
553         
554         return (G.is_break == TRUE);
555 }
556
557
558 /* ***************** GLOBAL UNDO *************** */
559
560 #define UNDO_DISK   0
561
562 typedef struct UndoElem {
563         struct UndoElem *next, *prev;
564         char str[FILE_MAX];
565         char name[BKE_UNDO_STR_MAX];
566         MemFile memfile;
567         uintptr_t undosize;
568 } UndoElem;
569
570 static ListBase undobase = {NULL, NULL};
571 static UndoElem *curundo = NULL;
572
573
574 static int read_undosave(bContext *C, UndoElem *uel)
575 {
576         char mainstr[sizeof(G.main->name)];
577         int success = 0, fileflags;
578         
579         /* This is needed so undoing/redoing doesn't crash with threaded previews going */
580         WM_jobs_kill_all_except(CTX_wm_manager(C), CTX_wm_screen(C));
581
582         BLI_strncpy(mainstr, G.main->name, sizeof(mainstr));    /* temporal store */
583
584         fileflags = G.fileflags;
585         G.fileflags |= G_FILE_NO_UI;
586
587         if (UNDO_DISK) 
588                 success = (BKE_read_file(C, uel->str, NULL) != BKE_READ_FILE_FAIL);
589         else
590                 success = BKE_read_file_from_memfile(C, &uel->memfile, NULL);
591
592         /* restore */
593         BLI_strncpy(G.main->name, mainstr, sizeof(G.main->name)); /* restore */
594         G.fileflags = fileflags;
595
596         if (success) {
597                 /* important not to update time here, else non keyed tranforms are lost */
598                 DAG_on_visible_update(G.main, FALSE);
599         }
600
601         return success;
602 }
603
604 /* name can be a dynamic string */
605 void BKE_write_undo(bContext *C, const char *name)
606 {
607         uintptr_t maxmem, totmem, memused;
608         int nr /*, success */ /* UNUSED */;
609         UndoElem *uel;
610         
611         if ((U.uiflag & USER_GLOBALUNDO) == 0) {
612                 return;
613         }
614
615         if (U.undosteps == 0) {
616                 return;
617         }
618         
619         /* remove all undos after (also when curundo == NULL) */
620         while (undobase.last != curundo) {
621                 uel = undobase.last;
622                 BLI_remlink(&undobase, uel);
623                 BLO_free_memfile(&uel->memfile);
624                 MEM_freeN(uel);
625         }
626         
627         /* make new */
628         curundo = uel = MEM_callocN(sizeof(UndoElem), "undo file");
629         BLI_strncpy(uel->name, name, sizeof(uel->name));
630         BLI_addtail(&undobase, uel);
631         
632         /* and limit amount to the maximum */
633         nr = 0;
634         uel = undobase.last;
635         while (uel) {
636                 nr++;
637                 if (nr == U.undosteps) break;
638                 uel = uel->prev;
639         }
640         if (uel) {
641                 while (undobase.first != uel) {
642                         UndoElem *first = undobase.first;
643                         BLI_remlink(&undobase, first);
644                         /* the merge is because of compression */
645                         BLO_merge_memfile(&first->memfile, &first->next->memfile);
646                         MEM_freeN(first);
647                 }
648         }
649
650
651         /* disk save version */
652         if (UNDO_DISK) {
653                 static int counter = 0;
654                 char filepath[FILE_MAX];
655                 char numstr[32];
656                 int fileflags = G.fileflags & ~(G_FILE_HISTORY); /* don't do file history on undo */
657
658                 /* calculate current filepath */
659                 counter++;
660                 counter = counter % U.undosteps;
661         
662                 BLI_snprintf(numstr, sizeof(numstr), "%d.blend", counter);
663                 BLI_make_file_string("/", filepath, BLI_temporary_dir(), numstr);
664         
665                 /* success = */ /* UNUSED */ BLO_write_file(CTX_data_main(C), filepath, fileflags, NULL, NULL);
666                 
667                 BLI_strncpy(curundo->str, filepath, sizeof(curundo->str));
668         }
669         else {
670                 MemFile *prevfile = NULL;
671                 
672                 if (curundo->prev) prevfile = &(curundo->prev->memfile);
673                 
674                 memused = MEM_get_memory_in_use();
675                 /* success = */ /* UNUSED */ BLO_write_file_mem(CTX_data_main(C), prevfile, &curundo->memfile, G.fileflags);
676                 curundo->undosize = MEM_get_memory_in_use() - memused;
677         }
678
679         if (U.undomemory != 0) {
680                 /* limit to maximum memory (afterwards, we can't know in advance) */
681                 totmem = 0;
682                 maxmem = ((uintptr_t)U.undomemory) * 1024 * 1024;
683
684                 /* keep at least two (original + other) */
685                 uel = undobase.last;
686                 while (uel && uel->prev) {
687                         totmem += uel->undosize;
688                         if (totmem > maxmem) break;
689                         uel = uel->prev;
690                 }
691
692                 if (uel) {
693                         if (uel->prev && uel->prev->prev)
694                                 uel = uel->prev;
695
696                         while (undobase.first != uel) {
697                                 UndoElem *first = undobase.first;
698                                 BLI_remlink(&undobase, first);
699                                 /* the merge is because of compression */
700                                 BLO_merge_memfile(&first->memfile, &first->next->memfile);
701                                 MEM_freeN(first);
702                         }
703                 }
704         }
705 }
706
707 /* 1 = an undo, -1 is a redo. we have to make sure 'curundo' remains at current situation */
708 void BKE_undo_step(bContext *C, int step)
709 {
710         
711         if (step == 0) {
712                 read_undosave(C, curundo);
713         }
714         else if (step == 1) {
715                 /* curundo should never be NULL, after restart or load file it should call undo_save */
716                 if (curundo == NULL || curundo->prev == NULL) {
717                         // XXX error("No undo available");
718                 }
719                 else {
720                         if (G.debug & G_DEBUG) printf("undo %s\n", curundo->name);
721                         curundo = curundo->prev;
722                         read_undosave(C, curundo);
723                 }
724         }
725         else {
726                 /* curundo has to remain current situation! */
727                 
728                 if (curundo == NULL || curundo->next == NULL) {
729                         // XXX error("No redo available");
730                 }
731                 else {
732                         read_undosave(C, curundo->next);
733                         curundo = curundo->next;
734                         if (G.debug & G_DEBUG) printf("redo %s\n", curundo->name);
735                 }
736         }
737 }
738
739 void BKE_reset_undo(void)
740 {
741         UndoElem *uel;
742         
743         uel = undobase.first;
744         while (uel) {
745                 BLO_free_memfile(&uel->memfile);
746                 uel = uel->next;
747         }
748         
749         BLI_freelistN(&undobase);
750         curundo = NULL;
751 }
752
753 /* based on index nr it does a restore */
754 void BKE_undo_number(bContext *C, int nr)
755 {
756         curundo = BLI_findlink(&undobase, nr);
757         BKE_undo_step(C, 0);
758 }
759
760 /* go back to the last occurance of name in stack */
761 void BKE_undo_name(bContext *C, const char *name)
762 {
763         UndoElem *uel = BLI_rfindstring(&undobase, name, offsetof(UndoElem, name));
764
765         if (uel && uel->prev) {
766                 curundo = uel->prev;
767                 BKE_undo_step(C, 0);
768         }
769 }
770
771 /* name optional */
772 int BKE_undo_valid(const char *name)
773 {
774         if (name) {
775                 UndoElem *uel = BLI_rfindstring(&undobase, name, offsetof(UndoElem, name));
776                 return uel && uel->prev;
777         }
778         
779         return undobase.last != undobase.first;
780 }
781
782 /* get name of undo item, return null if no item with this index */
783 /* if active pointer, set it to 1 if true */
784 const char *BKE_undo_get_name(int nr, int *active)
785 {
786         UndoElem *uel = BLI_findlink(&undobase, nr);
787         
788         if (active) *active = 0;
789         
790         if (uel) {
791                 if (active && uel == curundo)
792                         *active = 1;
793                 return uel->name;
794         }
795         return NULL;
796 }
797
798 /* saves .blend using undo buffer, returns 1 == success */
799 int BKE_undo_save_file(const char *filename)
800 {
801         UndoElem *uel;
802         MemFileChunk *chunk;
803         const int flag = O_BINARY + O_WRONLY + O_CREAT + O_TRUNC + O_EXCL;
804         int file;
805
806         if ((U.uiflag & USER_GLOBALUNDO) == 0) {
807                 return 0;
808         }
809
810         uel = curundo;
811         if (uel == NULL) {
812                 fprintf(stderr, "No undo buffer to save recovery file\n");
813                 return 0;
814         }
815
816         /* first try create the file, if it exists call without 'O_CREAT',
817          * to avoid writing to a symlink - use 'O_EXCL' (CVE-2008-1103) */
818         errno = 0;
819         file = BLI_open(filename, flag, 0666);
820         if (file < 0) {
821                 if (errno == EEXIST) {
822                         errno = 0;
823                         file = BLI_open(filename, flag & ~O_CREAT, 0666);
824                 }
825         }
826
827         if (file == -1) {
828                 fprintf(stderr, "Unable to save '%s': %s\n",
829                         filename, errno ? strerror(errno) : "Unknown error opening file");
830                 return 0;
831         }
832
833         for (chunk = uel->memfile.chunks.first; chunk; chunk = chunk->next) {
834                 if (write(file, chunk->buf, chunk->size) != chunk->size) {
835                         break;
836                 }
837         }
838         
839         close(file);
840         
841         if (chunk) {
842                 fprintf(stderr, "Unable to save '%s': %s\n",
843                         filename, errno ? strerror(errno) : "Unknown error writing file");
844                 return 0;
845         }
846         return 1;
847 }
848
849 /* sets curscene */
850 Main *BKE_undo_get_main(Scene **scene)
851 {
852         Main *mainp = NULL;
853         BlendFileData *bfd = BLO_read_from_memfile(G.main, G.main->name, &curundo->memfile, NULL);
854         
855         if (bfd) {
856                 mainp = bfd->main;
857                 if (scene)
858                         *scene = bfd->curscene;
859                 
860                 MEM_freeN(bfd);
861         }
862         
863         return mainp;
864 }
865
866 /* ************** copy paste .blend, partial saves ********** */
867
868 /* assumes data is in G.main */
869
870 void BKE_copybuffer_begin(void)
871 {
872         /* set all id flags to zero; */
873         flag_all_listbases_ids(LIB_NEED_EXPAND | LIB_DOIT, 0);
874 }
875
876 void BKE_copybuffer_tag_ID(ID *id)
877 {
878         id->flag |= LIB_NEED_EXPAND | LIB_DOIT;
879 }
880
881 static void copybuffer_doit(void *UNUSED(handle), Main *UNUSED(bmain), void *vid)
882 {
883         if (vid) {
884                 ID *id = vid;
885                 /* only tag for need-expand if not done, prevents eternal loops */
886                 if ((id->flag & LIB_DOIT) == 0)
887                         id->flag |= LIB_NEED_EXPAND | LIB_DOIT;
888         }
889 }
890
891 /* frees main in end */
892 int BKE_copybuffer_save(const char *filename, ReportList *reports)
893 {
894         Main *mainb = MEM_callocN(sizeof(Main), "copybuffer");
895         ListBase *lbarray[MAX_LIBARRAY], *fromarray[MAX_LIBARRAY];
896         int a, retval;
897         
898         BLO_main_expander(copybuffer_doit);
899         BLO_expand_main(NULL, G.main);
900         
901         /* move over all tagged blocks */
902         set_listbasepointers(G.main, fromarray);
903         a = set_listbasepointers(mainb, lbarray);
904         while (a--) {
905                 ID *id, *nextid;
906                 ListBase *lb1 = lbarray[a], *lb2 = fromarray[a];
907                 
908                 for (id = lb2->first; id; id = nextid) {
909                         nextid = id->next;
910                         if (id->flag & LIB_DOIT) {
911                                 BLI_remlink(lb2, id);
912                                 BLI_addtail(lb1, id);
913                         }
914                 }
915         }
916         
917         
918         /* save the buffer */
919         retval = BLO_write_file(mainb, filename, 0, reports, NULL);
920         
921         /* move back the main, now sorted again */
922         set_listbasepointers(G.main, lbarray);
923         a = set_listbasepointers(mainb, fromarray);
924         while (a--) {
925                 ID *id;
926                 ListBase *lb1 = lbarray[a], *lb2 = fromarray[a];
927                 
928                 while (lb2->first) {
929                         id = lb2->first;
930                         BLI_remlink(lb2, id);
931                         BLI_addtail(lb1, id);
932                         id_sort_by_name(lb1, id);
933                 }
934         }
935         
936         MEM_freeN(mainb);
937         
938         /* set id flag to zero; */
939         flag_all_listbases_ids(LIB_NEED_EXPAND | LIB_DOIT, 0);
940         
941         return retval;
942 }
943
944 /* return success (1) */
945 int BKE_copybuffer_paste(bContext *C, const char *libname, ReportList *reports)
946 {
947         Main *bmain = CTX_data_main(C);
948         Scene *scene = CTX_data_scene(C);
949         Main *mainl = NULL;
950         Library *lib;
951         BlendHandle *bh;
952                 
953         bh = BLO_blendhandle_from_file(libname, reports);
954         
955         if (bh == NULL) {
956                 /* error reports will have been made by BLO_blendhandle_from_file() */
957                 return 0;
958         }
959
960         BKE_scene_base_deselect_all(scene);
961         
962         /* tag everything, all untagged data can be made local
963          * its also generally useful to know what is new
964          *
965          * take extra care flag_all_listbases_ids(LIB_LINK_TAG, 0) is called after! */
966         flag_all_listbases_ids(LIB_PRE_EXISTING, 1);
967         
968         /* here appending/linking starts */
969         mainl = BLO_library_append_begin(bmain, &bh, libname);
970         
971         BLO_library_append_all(mainl, bh);
972
973         BLO_library_append_end(C, mainl, &bh, 0, 0);
974         
975         /* mark all library linked objects to be updated */
976         recalc_all_library_objects(bmain);
977         IMB_colormanagement_check_file_config(bmain);
978         
979         /* append, rather than linking */
980         lib = BLI_findstring(&bmain->library, libname, offsetof(Library, filepath));
981         BKE_library_make_local(bmain, lib, true);
982         
983         /* important we unset, otherwise these object wont
984          * link into other scenes from this blend file */
985         flag_all_listbases_ids(LIB_PRE_EXISTING, 0);
986         
987         /* recreate dependency graph to include new objects */
988         DAG_relations_tag_update(bmain);
989         
990         BLO_blendhandle_close(bh);
991         /* remove library... */
992         
993         return 1;
994 }