add a segfault handler that writes out the info log into a crash file alongside the...
[blender.git] / source / blender / blenloader / intern / runtime.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
29 /**
30  * \file runtime.c
31  * \brief This file handles the loading of .blend files embedded in runtimes
32  * \ingroup blenloader
33  */
34
35 #include <stdio.h>
36 #include <stdlib.h>
37 #include <string.h>
38 #include <fcntl.h>
39 #include <errno.h>
40
41 #ifdef WIN32
42 #  include <io.h>       // read, open
43 #  include "BLI_winstuff.h"
44 #else // ! WIN32
45 #  include <unistd.h>       // read
46 #endif
47
48 #include "BLO_readfile.h"
49 #include "BLO_runtime.h"
50
51 #include "BLI_blenlib.h"
52 #include "BLI_utildefines.h"
53
54 #include "BKE_blender.h"
55 #include "BKE_report.h"
56
57 /* Runtime reading */
58
59 static int handle_read_msb_int(int handle)
60 {
61         unsigned char buf[4];
62
63         if (read(handle, buf, 4) != 4)
64                 return -1;
65
66         return (buf[0] << 24) + (buf[1] << 16) + (buf[2] << 8) + (buf[3] << 0);
67 }
68
69 int BLO_is_a_runtime(const char *path)
70 {
71         int res = 0, fd = BLI_open(path, O_BINARY | O_RDONLY, 0);
72         int datastart;
73         char buf[8];
74
75         if (fd == -1)
76                 goto cleanup;
77         
78         lseek(fd, -12, SEEK_END);
79         
80         datastart = handle_read_msb_int(fd);
81
82         if (datastart == -1)
83                 goto cleanup;
84         else if (read(fd, buf, 8) != 8)
85                 goto cleanup;
86         else if (memcmp(buf, "BRUNTIME", 8) != 0)
87                 goto cleanup;
88         else
89                 res = 1;
90
91 cleanup:
92         if (fd != -1)
93                 close(fd);
94
95         return res;
96 }
97
98 BlendFileData *BLO_read_runtime(const char *path, ReportList *reports)
99 {
100         BlendFileData *bfd = NULL;
101         size_t actualsize;
102         int fd, datastart;
103         char buf[8];
104
105         fd = BLI_open(path, O_BINARY | O_RDONLY, 0);
106
107         if (fd == -1) {
108                 BKE_reportf(reports, RPT_ERROR, "Unable to open '%s': %s", path, strerror(errno));
109                 goto cleanup;
110         }
111         
112         actualsize = BLI_file_descriptor_size(fd);
113
114         lseek(fd, -12, SEEK_END);
115
116         datastart = handle_read_msb_int(fd);
117
118         if (datastart == -1) {
119                 BKE_reportf(reports, RPT_ERROR, "Unable to read '%s' (problem seeking)", path);
120                 goto cleanup;
121         }
122         else if (read(fd, buf, 8) != 8) {
123                 BKE_reportf(reports, RPT_ERROR, "Unable to read '%s' (truncated header)", path);
124                 goto cleanup;
125         }
126         else if (memcmp(buf, "BRUNTIME", 8) != 0) {
127                 BKE_reportf(reports, RPT_ERROR, "Unable to read '%s' (not a blend file)", path);
128                 goto cleanup;
129         }
130         else {
131                 //printf("starting to read runtime from %s at datastart %d\n", path, datastart);
132                 lseek(fd, datastart, SEEK_SET);
133                 bfd = blo_read_blendafterruntime(fd, path, actualsize - datastart, reports);
134                 fd = -1; // file was closed in blo_read_blendafterruntime()
135         }
136         
137 cleanup:
138         if (fd != -1)
139                 close(fd);
140         
141         return bfd;
142 }
143