merge with/from trunk at r35190
[blender.git] / source / blender / readblenfile / intern / BLO_readblenfile.c
1 /*
2  * $Id$
3  *
4  * ***** BEGIN GPL LICENSE BLOCK *****
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version 2
9  * of the License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software Foundation,
18  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19  *
20  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
21  * All rights reserved.
22  *
23  * The Original Code is: all of this file.
24  *
25  * Contributor(s): none yet.
26  *
27  * ***** END GPL LICENSE BLOCK *****
28  *
29  */
30 /**
31  * \file BLO_readblenfile.c
32  * \brief This file handles the loading if .blend files
33  * \ingroup mainmodule
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_readblenfile.h"
50
51 #include "BKE_blender.h"
52 #include "BKE_report.h"
53
54 #include "BLI_blenlib.h"
55
56 /** Magic number for the file header */
57 const char *headerMagic = "BLENDFI";
58
59 /**
60  * \brief Set the version number into the array.
61  *
62  * version contains the integer number of the version
63  * i.e. 227
64  * array[1] gets set to the div of the number by 100 i.e. 2
65  * array[2] gets the remainder i.e. 27
66  */
67 void BLO_setversionnumber(char array[4], int version)
68 {
69         memset(array, 0, sizeof(char)*4);
70
71         array[1] = version / 100;
72         array[2] = version % 100;
73 }
74
75 /**
76  * Sets version number using BLENDER_VERSION
77  * Function that calls the setversionnumber(char[],int) with 
78  * the BLENDER_VERSION constant and sets the resultant array
79  * with the version parts.  
80  * see BLO_setversionnumber(char[],int).
81  */
82 void BLO_setcurrentversionnumber(char array[4])
83 {
84         BLO_setversionnumber(array, BLENDER_VERSION);
85 }
86
87 #ifndef O_BINARY
88 #define O_BINARY 0
89 #endif
90
91 /* Runtime reading */
92
93 static int handle_read_msb_int(int handle) {
94         unsigned char buf[4];
95
96         if (read(handle, buf, 4)!=4)
97                 return -1;
98         else
99                 return (buf[0]<<24) + (buf[1]<<16) + (buf[2]<<8) + (buf[3]<<0);
100 }
101
102 int blo_is_a_runtime(char *path) {
103         int res= 0, fd= open(path, O_BINARY|O_RDONLY, 0);
104         int datastart;
105         char buf[8];
106
107         if (fd==-1)
108                 goto cleanup;
109         
110         lseek(fd, -12, SEEK_END);
111         
112         datastart= handle_read_msb_int(fd);
113         if (datastart==-1)
114                 goto cleanup;
115         else if (read(fd, buf, 8)!=8)
116                 goto cleanup;
117         else if (memcmp(buf, "BRUNTIME", 8)!=0)
118                 goto cleanup;
119         else
120                 res= 1;
121
122 cleanup:
123         if (fd!=-1)
124                 close(fd);
125
126         return res;     
127 }
128
129 BlendFileData *
130 blo_read_runtime(
131         char *path, 
132         ReportList *reports)
133 {
134         BlendFileData *bfd= NULL;
135         size_t actualsize;
136         int fd, datastart;
137         char buf[8];
138
139         fd= open(path, O_BINARY|O_RDONLY, 0);
140         if (fd==-1) {
141                 BKE_reportf(reports, RPT_ERROR, "Unable to open \"%s\": %s.", path, strerror(errno));
142                 goto cleanup;
143         }
144         
145         actualsize= BLI_filesize(fd);
146
147         lseek(fd, -12, SEEK_END);
148
149         datastart= handle_read_msb_int(fd);
150         if (datastart==-1) {
151                 BKE_reportf(reports, RPT_ERROR, "Unable to read  \"%s\" (problem seeking)", path);
152                 goto cleanup;
153         } else if (read(fd, buf, 8)!=8) {
154                 BKE_reportf(reports, RPT_ERROR, "Unable to read  \"%s\" (truncated header)", path);
155                 goto cleanup;
156         } else if (memcmp(buf, "BRUNTIME", 8)!=0) {
157                 BKE_reportf(reports, RPT_ERROR, "Unable to read  \"%s\" (not a blend file)", path);
158                 goto cleanup;
159         } else {        
160                 //printf("starting to read runtime from %s at datastart %d\n", path, datastart);
161                 lseek(fd, datastart, SEEK_SET);
162                 bfd = blo_read_blendafterruntime(fd, path, actualsize-datastart, reports);
163                 fd= -1; // file was closed in blo_read_blendafterruntime()
164         }
165         
166 cleanup:
167         if (fd!=-1)
168                 close(fd);
169         
170         return bfd;
171 }