Filling in branch from trunk
[blender.git] / source / blender / readblenfile / intern / BLO_readblenfile.c
1 /*
2  * $Id$
3  *
4  * ***** BEGIN GPL/BL DUAL 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. The Blender
10  * Foundation also sells licenses for use in proprietary software under
11  * the Blender License.  See http://www.blender.org/BL/ for information
12  * about this.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software Foundation,
21  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22  *
23  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
24  * All rights reserved.
25  *
26  * The Original Code is: all of this file.
27  *
28  * Contributor(s): none yet.
29  *
30  * ***** END GPL/BL DUAL LICENSE BLOCK *****
31  *
32  */
33 /**
34  * \file BLO_readblenfile.c
35  * \brief This file handles the loading if .blend files
36  * \ingroup mainmodule
37  */
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <string.h>
41 #include <fcntl.h>
42
43 #ifdef HAVE_CONFIG_H
44 #include <config.h>
45 #endif
46
47 #ifdef WIN32
48 #include <io.h>         // read, open
49 #else // ! WIN32
50 #include <unistd.h>             // read
51 #endif
52
53 #include "BLO_readfile.h"
54 #include "BLO_readblenfile.h"
55
56 #include "BKE_blender.h"
57
58 #include "BLI_blenlib.h"
59
60 /** Magic number for the file header */
61 char *headerMagic = "BLENDFI";
62
63 /**
64  * \brief Set the version number into the array.
65  *
66  * version contains the integer number of the version
67  * i.e. 227
68  * array[1] gets set to the div of the number by 100 i.e. 2
69  * array[2] gets the remainder i.e. 27
70  */
71 void BLO_setversionnumber(char array[4], int version)
72 {
73         memset(array, 0, sizeof(array));
74
75         array[1] = version / 100;
76         array[2] = version % 100;
77 }
78
79 /**
80  * Sets version number using BLENDER_VERSION
81  * Function that calls the setversionnumber(char[],int) with 
82  * the BLENDER_VERSION constant and sets the resultant array
83  * with the version parts.  
84  * see BLO_setversionnumber(char[],int).
85  */
86 void BLO_setcurrentversionnumber(char array[4])
87 {
88         BLO_setversionnumber(array, BLENDER_VERSION);
89 }
90
91 #ifndef O_BINARY
92 #define O_BINARY 0
93 #endif
94
95 /* Runtime reading */
96
97 static int handle_read_msb_int(int handle) {
98         unsigned char buf[4];
99
100         if (read(handle, buf, 4)!=4)
101                 return -1;
102         else
103                 return (buf[0]<<24) + (buf[1]<<16) + (buf[2]<<8) + (buf[3]<<0);
104 }
105
106 int blo_is_a_runtime(char *path) {
107         int res= 0, fd= open(path, O_BINARY|O_RDONLY, 0);
108         int datastart;
109         char buf[8];
110
111         if (fd==-1)
112                 goto cleanup;
113         
114         lseek(fd, -12, SEEK_END);
115         
116         datastart= handle_read_msb_int(fd);
117         if (datastart==-1)
118                 goto cleanup;
119         else if (read(fd, buf, 8)!=8)
120                 goto cleanup;
121         else if (memcmp(buf, "BRUNTIME", 8)!=0)
122                 goto cleanup;
123         else
124                 res= 1;
125
126 cleanup:
127         if (fd!=-1)
128                 close(fd);
129
130         return res;     
131 }
132
133 BlendFileData *
134 blo_read_runtime(
135         char *path, 
136         BlendReadError *error_r) 
137 {
138         BlendFileData *bfd= NULL;
139         int fd, actualsize, datastart;
140         char buf[8];
141
142         fd= open(path, O_BINARY|O_RDONLY, 0);
143         if (fd==-1) {
144                 *error_r= BRE_UNABLE_TO_OPEN;
145                 goto cleanup;
146         }
147         
148         actualsize= BLI_filesize(fd);
149
150         lseek(fd, -12, SEEK_END);
151
152         datastart= handle_read_msb_int(fd);
153         if (datastart==-1) {
154                 *error_r= BRE_UNABLE_TO_READ;
155                 goto cleanup;
156         } else if (read(fd, buf, 8)!=8) {
157                 *error_r= BRE_UNABLE_TO_READ;
158                 goto cleanup;
159         } else if (memcmp(buf, "BRUNTIME", 8)!=0) {
160                 *error_r= BRE_NOT_A_BLEND;
161                 goto cleanup;
162         } else {        
163                 //printf("starting to read runtime from %s at datastart %d\n", path, datastart);
164                 lseek(fd, datastart, SEEK_SET);
165                 bfd = blo_read_blendafterruntime(fd, actualsize-datastart, error_r);
166                 fd= -1; // file was closed in blo_read_blendafterruntime()
167         }
168         
169 cleanup:
170         if (fd!=-1)
171                 close(fd);
172         
173         return bfd;
174 }
175