added CVAR_SAVE and CVAR_NOTIFY flags to cvar_t structure (at the beginning), updated...
[divverent/darkplaces.git] / model_shared.c
1 /*
2 Copyright (C) 1996-1997 Id Software, Inc.
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.  
12
13 See the GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
18
19 */
20 // models.c -- model loading and caching
21
22 // models are the only shared resource between a client and server running
23 // on the same machine.
24
25 #include "quakedef.h"
26
27 model_t *loadmodel;
28 char    loadname[32];   // for hunk tags
29
30 // LordHavoc: increased from 512 to 2048
31 #define MAX_MOD_KNOWN   2048
32 model_t mod_known[MAX_MOD_KNOWN];
33 int             mod_numknown;
34
35 /*
36 ===============
37 Mod_Init
38 ===============
39 */
40 void Mod_Init (void)
41 {
42         Mod_BrushInit();
43         Mod_AliasInit();
44         Mod_SpriteInit();
45 }
46
47 /*
48 ===============
49 Mod_Init
50
51 Caches the data if needed
52 ===============
53 */
54 void *Mod_Extradata (model_t *mod)
55 {
56         void    *r;
57         
58         r = Cache_Check (&mod->cache);
59         if (r)
60                 return r;
61
62         Mod_LoadModel (mod, true);
63         
64         if (!mod->cache.data)
65                 Host_Error ("Mod_Extradata: caching failed");
66         return mod->cache.data;
67 }
68
69 /*
70 ===================
71 Mod_ClearAll
72 ===================
73 */
74 void Mod_ClearAll (void)
75 {
76         int             i;
77         model_t *mod;
78         
79         for (i=0 , mod=mod_known ; i<mod_numknown ; i++, mod++)
80                 if (!mod->cachesize)
81                         mod->needload = true;
82 }
83
84 /*
85 ==================
86 Mod_FindName
87
88 ==================
89 */
90 model_t *Mod_FindName (char *name)
91 {
92         int             i;
93         model_t *mod;
94         
95         if (!name[0])
96                 Host_Error ("Mod_ForName: NULL name");
97                 
98 //
99 // search the currently loaded models
100 //
101         for (i=0 , mod=mod_known ; i<mod_numknown ; i++, mod++)
102                 if (!strcmp (mod->name, name) )
103                         break;
104                         
105         if (i == mod_numknown)
106         {
107                 if (mod_numknown == MAX_MOD_KNOWN)
108                         Host_Error ("mod_numknown == MAX_MOD_KNOWN");
109                 strcpy (mod->name, name);
110                 mod->needload = true;
111                 mod_numknown++;
112         }
113
114         return mod;
115 }
116
117 /*
118 ==================
119 Mod_TouchModel
120
121 ==================
122 */
123 void Mod_TouchModel (char *name)
124 {
125         model_t *mod;
126         
127         mod = Mod_FindName (name);
128         
129         if (!mod->needload)
130                 if (mod->cachesize)
131                         Cache_Check (&mod->cache);
132 }
133
134 /*
135 ==================
136 Mod_LoadModel
137
138 Loads a model into the cache
139 ==================
140 */
141 model_t *Mod_LoadModel (model_t *mod, qboolean crash)
142 {
143         void    *d;
144         unsigned *buf;
145
146         if (!mod->needload)
147         {
148                 if (mod->cachesize)
149                 {
150                         d = Cache_Check (&mod->cache);
151                         if (d)
152                                 return mod;
153                 }
154                 else
155                         return mod;             // not cached at all
156         }
157
158         Con_DPrintf("loading model %s\n", mod->name);
159
160 // load the file
161         buf = (unsigned *)COM_LoadMallocFile (mod->name, false);
162         if (!buf)
163         {
164                 if (crash)
165                         Host_Error ("Mod_NumForName: %s not found", mod->name); // LordHavoc: Sys_Error was *ANNOYING*
166                 return NULL;
167         }
168         
169 // allocate a new model
170 //      COM_FileBase (mod->name, loadname);
171         strcpy(loadname, mod->name);
172         
173         loadmodel = mod;
174
175 // call the apropriate loader
176         mod->needload = false;
177         
178         switch (LittleLong(*(unsigned *)buf))
179         {
180         case IDPOLYHEADER:
181                 Mod_LoadAliasModel (mod, buf);
182                 break;
183
184         case MD2IDALIASHEADER: // LordHavoc: added Quake2 model support
185                 Mod_LoadQ2AliasModel (mod, buf);
186                 break;
187
188         case (('O'<<24)+('M'<<16)+('Y'<<8)+'Z'):
189                 Mod_LoadZymoticModel(mod, buf);
190                 break;
191                 
192         case IDSPRITEHEADER:
193                 Mod_LoadSpriteModel (mod, buf);
194                 break;
195         
196         default:
197                 Mod_LoadBrushModel (mod, buf);
198                 break;
199         }
200         qfree(buf);
201
202         return mod;
203 }
204
205 /*
206 ==================
207 Mod_ForName
208
209 Loads in a model for the given name
210 ==================
211 */
212 model_t *Mod_ForName (char *name, qboolean crash)
213 {
214         model_t *mod;
215         
216         mod = Mod_FindName (name);
217         
218         return Mod_LoadModel (mod, crash);
219 }
220
221 byte    *mod_base;
222
223 /*
224 =================
225 RadiusFromBounds
226 =================
227 */
228 float RadiusFromBounds (vec3_t mins, vec3_t maxs)
229 {
230         int             i;
231         vec3_t  corner;
232
233         for (i=0 ; i<3 ; i++)
234                 corner[i] = fabs(mins[i]) > fabs(maxs[i]) ? fabs(mins[i]) : fabs(maxs[i]);
235
236         return Length (corner);
237 }
238
239 //=============================================================================
240
241 /*
242 ================
243 Mod_Print
244 ================
245 */
246 void Mod_Print (void)
247 {
248         int             i;
249         model_t *mod;
250
251         Con_Printf ("Cached models:\n");
252         for (i=0, mod=mod_known ; i < mod_numknown ; i++, mod++)
253         {
254                 Con_Printf ("%4iK : %8p : %s\n", (mod->cachesize + 1023) / 1024, mod->cache.data, mod->name);
255         }
256 }
257
258