]> icculus.org git repositories - divverent/darkplaces.git/blob - mathlib.c
fix viewmodel crash
[divverent/darkplaces.git] / mathlib.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 // mathlib.c -- math primitives
21
22 #include <math.h>
23 #include "quakedef.h"
24
25 vec3_t vec3_origin = {0,0,0};
26 float ixtable[4096];
27
28 /*-----------------------------------------------------------------*/
29
30 float m_bytenormals[NUMVERTEXNORMALS][3] =
31 {
32 {-0.525731, 0.000000, 0.850651}, {-0.442863, 0.238856, 0.864188}, 
33 {-0.295242, 0.000000, 0.955423}, {-0.309017, 0.500000, 0.809017}, 
34 {-0.162460, 0.262866, 0.951056}, {0.000000, 0.000000, 1.000000}, 
35 {0.000000, 0.850651, 0.525731}, {-0.147621, 0.716567, 0.681718}, 
36 {0.147621, 0.716567, 0.681718}, {0.000000, 0.525731, 0.850651}, 
37 {0.309017, 0.500000, 0.809017}, {0.525731, 0.000000, 0.850651}, 
38 {0.295242, 0.000000, 0.955423}, {0.442863, 0.238856, 0.864188}, 
39 {0.162460, 0.262866, 0.951056}, {-0.681718, 0.147621, 0.716567}, 
40 {-0.809017, 0.309017, 0.500000}, {-0.587785, 0.425325, 0.688191}, 
41 {-0.850651, 0.525731, 0.000000}, {-0.864188, 0.442863, 0.238856},
42 {-0.716567, 0.681718, 0.147621}, {-0.688191, 0.587785, 0.425325}, 
43 {-0.500000, 0.809017, 0.309017}, {-0.238856, 0.864188, 0.442863}, 
44 {-0.425325, 0.688191, 0.587785}, {-0.716567, 0.681718, -0.147621}, 
45 {-0.500000, 0.809017, -0.309017}, {-0.525731, 0.850651, 0.000000}, 
46 {0.000000, 0.850651, -0.525731}, {-0.238856, 0.864188, -0.442863},
47 {0.000000, 0.955423, -0.295242}, {-0.262866, 0.951056, -0.162460}, 
48 {0.000000, 1.000000, 0.000000}, {0.000000, 0.955423, 0.295242},
49 {-0.262866, 0.951056, 0.162460}, {0.238856, 0.864188, 0.442863}, 
50 {0.262866, 0.951056, 0.162460}, {0.500000, 0.809017, 0.309017}, 
51 {0.238856, 0.864188, -0.442863}, {0.262866, 0.951056, -0.162460}, 
52 {0.500000, 0.809017, -0.309017}, {0.850651, 0.525731, 0.000000}, 
53 {0.716567, 0.681718, 0.147621}, {0.716567, 0.681718, -0.147621}, 
54 {0.525731, 0.850651, 0.000000}, {0.425325, 0.688191, 0.587785}, 
55 {0.864188, 0.442863, 0.238856}, {0.688191, 0.587785, 0.425325}, 
56 {0.809017, 0.309017, 0.500000}, {0.681718, 0.147621, 0.716567},
57 {0.587785, 0.425325, 0.688191}, {0.955423, 0.295242, 0.000000}, 
58 {1.000000, 0.000000, 0.000000}, {0.951056, 0.162460, 0.262866},
59 {0.850651, -0.525731, 0.000000}, {0.955423, -0.295242, 0.000000}, 
60 {0.864188, -0.442863, 0.238856}, {0.951056, -0.162460, 0.262866}, 
61 {0.809017, -0.309017, 0.500000}, {0.681718, -0.147621, 0.716567},
62 {0.850651, 0.000000, 0.525731}, {0.864188, 0.442863, -0.238856},
63 {0.809017, 0.309017, -0.500000}, {0.951056, 0.162460, -0.262866}, 
64 {0.525731, 0.000000, -0.850651}, {0.681718, 0.147621, -0.716567}, 
65 {0.681718, -0.147621, -0.716567}, {0.850651, 0.000000, -0.525731},
66 {0.809017, -0.309017, -0.500000}, {0.864188, -0.442863, -0.238856}, 
67 {0.951056, -0.162460, -0.262866}, {0.147621, 0.716567, -0.681718}, 
68 {0.309017, 0.500000, -0.809017}, {0.425325, 0.688191, -0.587785}, 
69 {0.442863, 0.238856, -0.864188}, {0.587785, 0.425325, -0.688191}, 
70 {0.688191, 0.587785, -0.425325}, {-0.147621, 0.716567, -0.681718}, 
71 {-0.309017, 0.500000, -0.809017}, {0.000000, 0.525731, -0.850651},
72 {-0.525731, 0.000000, -0.850651}, {-0.442863, 0.238856, -0.864188},
73 {-0.295242, 0.000000, -0.955423}, {-0.162460, 0.262866, -0.951056}, 
74 {0.000000, 0.000000, -1.000000}, {0.295242, 0.000000, -0.955423}, 
75 {0.162460, 0.262866, -0.951056}, {-0.442863, -0.238856, -0.864188}, 
76 {-0.309017, -0.500000, -0.809017}, {-0.162460, -0.262866, -0.951056}, 
77 {0.000000, -0.850651, -0.525731}, {-0.147621, -0.716567, -0.681718}, 
78 {0.147621, -0.716567, -0.681718}, {0.000000, -0.525731, -0.850651}, 
79 {0.309017, -0.500000, -0.809017}, {0.442863, -0.238856, -0.864188}, 
80 {0.162460, -0.262866, -0.951056}, {0.238856, -0.864188, -0.442863}, 
81 {0.500000, -0.809017, -0.309017}, {0.425325, -0.688191, -0.587785}, 
82 {0.716567, -0.681718, -0.147621}, {0.688191, -0.587785, -0.425325}, 
83 {0.587785, -0.425325, -0.688191}, {0.000000, -0.955423, -0.295242},
84 {0.000000, -1.000000, 0.000000}, {0.262866, -0.951056, -0.162460}, 
85 {0.000000, -0.850651, 0.525731}, {0.000000, -0.955423, 0.295242}, 
86 {0.238856, -0.864188, 0.442863}, {0.262866, -0.951056, 0.162460}, 
87 {0.500000, -0.809017, 0.309017}, {0.716567, -0.681718, 0.147621}, 
88 {0.525731, -0.850651, 0.000000}, {-0.238856, -0.864188, -0.442863}, 
89 {-0.500000, -0.809017, -0.309017}, {-0.262866, -0.951056, -0.162460}, 
90 {-0.850651, -0.525731, 0.000000}, {-0.716567, -0.681718, -0.147621},
91 {-0.716567, -0.681718, 0.147621}, {-0.525731, -0.850651, 0.000000}, 
92 {-0.500000, -0.809017, 0.309017}, {-0.238856, -0.864188, 0.442863},
93 {-0.262866, -0.951056, 0.162460}, {-0.864188, -0.442863, 0.238856},
94 {-0.809017, -0.309017, 0.500000}, {-0.688191, -0.587785, 0.425325},
95 {-0.681718, -0.147621, 0.716567}, {-0.442863, -0.238856, 0.864188},
96 {-0.587785, -0.425325, 0.688191}, {-0.309017, -0.500000, 0.809017},
97 {-0.147621, -0.716567, 0.681718}, {-0.425325, -0.688191, 0.587785},
98 {-0.162460, -0.262866, 0.951056}, {0.442863, -0.238856, 0.864188},
99 {0.162460, -0.262866, 0.951056}, {0.309017, -0.500000, 0.809017},
100 {0.147621, -0.716567, 0.681718}, {0.000000, -0.525731, 0.850651},
101 {0.425325, -0.688191, 0.587785}, {0.587785, -0.425325, 0.688191},
102 {0.688191, -0.587785, 0.425325}, {-0.955423, 0.295242, 0.000000},
103 {-0.951056, 0.162460, 0.262866}, {-1.000000, 0.000000, 0.000000},
104 {-0.850651, 0.000000, 0.525731}, {-0.955423, -0.295242, 0.000000},
105 {-0.951056, -0.162460, 0.262866}, {-0.864188, 0.442863, -0.238856},
106 {-0.951056, 0.162460, -0.262866}, {-0.809017, 0.309017, -0.500000},
107 {-0.864188, -0.442863, -0.238856}, {-0.951056, -0.162460, -0.262866},
108 {-0.809017, -0.309017, -0.500000}, {-0.681718, 0.147621, -0.716567},
109 {-0.681718, -0.147621, -0.716567}, {-0.850651, 0.000000, -0.525731},
110 {-0.688191, 0.587785, -0.425325}, {-0.587785, 0.425325, -0.688191},
111 {-0.425325, 0.688191, -0.587785}, {-0.425325, -0.688191, -0.587785},
112 {-0.587785, -0.425325, -0.688191}, {-0.688191, -0.587785, -0.425325},
113 };
114
115 qbyte NormalToByte(const vec3_t n)
116 {
117         int i, best;
118         float bestdistance, distance;
119
120         best = 0;
121         bestdistance = DotProduct (n, m_bytenormals[0]);
122         for (i = 1;i < NUMVERTEXNORMALS;i++)
123         {
124                 distance = DotProduct (n, m_bytenormals[i]);
125                 if (distance > bestdistance)
126                 {
127                         bestdistance = distance;
128                         best = i;
129                 }
130         }
131         return best;
132 }
133
134 // note: uses byte partly to force unsigned for the validity check
135 void ByteToNormal(qbyte num, vec3_t n)
136 {
137         if (num < NUMVERTEXNORMALS)
138                 VectorCopy(m_bytenormals[num], n);
139         else
140                 VectorClear(n); // FIXME: complain?
141 }
142
143 float Q_RSqrt(float number)
144 {
145         float y;
146
147         if (number == 0.0f)
148                 return 0.0f;
149
150         *((int *)&y) = 0x5f3759df - ((* (int *) &number) >> 1);
151         return y * (1.5f - (number * 0.5f * y * y));
152 }
153
154
155 // assumes "src" is normalized
156 void PerpendicularVector( vec3_t dst, const vec3_t src )
157 {
158         // LordHavoc: optimized to death and beyond
159         int pos;
160         float minelem;
161
162         if (src[0])
163         {
164                 dst[0] = 0;
165                 if (src[1])
166                 {
167                         dst[1] = 0;
168                         if (src[2])
169                         {
170                                 dst[2] = 0;
171                                 pos = 0;
172                                 minelem = fabs(src[0]);
173                                 if (fabs(src[1]) < minelem)
174                                 {
175                                         pos = 1;
176                                         minelem = fabs(src[1]);
177                                 }
178                                 if (fabs(src[2]) < minelem)
179                                         pos = 2;
180
181                                 dst[pos] = 1;
182                                 dst[0] -= src[pos] * src[0];
183                                 dst[1] -= src[pos] * src[1];
184                                 dst[2] -= src[pos] * src[2];
185
186                                 // normalize the result
187                                 VectorNormalize(dst);
188                         }
189                         else
190                                 dst[2] = 1;
191                 }
192                 else
193                 {
194                         dst[1] = 1;
195                         dst[2] = 0;
196                 }
197         }
198         else
199         {
200                 dst[0] = 1;
201                 dst[1] = 0;
202                 dst[2] = 0;
203         }
204 }
205
206
207 // LordHavoc: like AngleVectors, but taking a forward vector instead of angles, useful!
208 void VectorVectors(const vec3_t forward, vec3_t right, vec3_t up)
209 {
210         float d;
211
212         right[0] = forward[2];
213         right[1] = -forward[0];
214         right[2] = forward[1];
215
216         d = DotProduct(forward, right);
217         right[0] -= d * forward[0];
218         right[1] -= d * forward[1];
219         right[2] -= d * forward[2];
220         VectorNormalizeFast(right);
221         CrossProduct(right, forward, up);
222 }
223
224 void VectorVectorsDouble(const double *forward, double *right, double *up)
225 {
226         double d;
227
228         right[0] = forward[2];
229         right[1] = -forward[0];
230         right[2] = forward[1];
231
232         d = DotProduct(forward, right);
233         right[0] -= d * forward[0];
234         right[1] -= d * forward[1];
235         right[2] -= d * forward[2];
236         VectorNormalize(right);
237         CrossProduct(right, forward, up);
238 }
239
240 void RotatePointAroundVector( vec3_t dst, const vec3_t dir, const vec3_t point, float degrees )
241 {
242         float t0, t1;
243         float angle, c, s;
244         vec3_t vr, vu, vf;
245
246         angle = DEG2RAD(degrees);
247
248         c = cos(angle);
249         s = sin(angle);
250
251         vf[0] = dir[0];
252         vf[1] = dir[1];
253         vf[2] = dir[2];
254
255         VectorVectors(vf, vr, vu);
256
257         t0 = vr[0] *  c + vu[0] * -s;
258         t1 = vr[0] *  s + vu[0] *  c;
259         dst[0] = (t0 * vr[0] + t1 * vu[0] + vf[0] * vf[0]) * point[0]
260                + (t0 * vr[1] + t1 * vu[1] + vf[0] * vf[1]) * point[1]
261                + (t0 * vr[2] + t1 * vu[2] + vf[0] * vf[2]) * point[2];
262
263         t0 = vr[1] *  c + vu[1] * -s;
264         t1 = vr[1] *  s + vu[1] *  c;
265         dst[1] = (t0 * vr[0] + t1 * vu[0] + vf[1] * vf[0]) * point[0]
266                + (t0 * vr[1] + t1 * vu[1] + vf[1] * vf[1]) * point[1]
267                + (t0 * vr[2] + t1 * vu[2] + vf[1] * vf[2]) * point[2];
268
269         t0 = vr[2] *  c + vu[2] * -s;
270         t1 = vr[2] *  s + vu[2] *  c;
271         dst[2] = (t0 * vr[0] + t1 * vu[0] + vf[2] * vf[0]) * point[0]
272                + (t0 * vr[1] + t1 * vu[1] + vf[2] * vf[1]) * point[1]
273                + (t0 * vr[2] + t1 * vu[2] + vf[2] * vf[2]) * point[2];
274 }
275
276 /*-----------------------------------------------------------------*/
277
278
279 void BoxOnPlaneSideClassify(mplane_t *p)
280 {
281         p->signbits = 0;
282         if (p->normal[0] < 0) // 1
283                 p->signbits |= 1;
284         if (p->normal[1] < 0) // 2
285                 p->signbits |= 2;
286         if (p->normal[2] < 0) // 4
287                 p->signbits |= 4;
288 }
289
290 void PlaneClassify(mplane_t *p)
291 {
292         if (p->normal[0] == 1)
293                 p->type = 0;
294         else if (p->normal[1] == 1)
295                 p->type = 1;
296         else if (p->normal[2] == 1)
297                 p->type = 2;
298         else
299                 p->type = 3;
300         BoxOnPlaneSideClassify(p);
301 }
302
303 int BoxOnPlaneSide (const vec3_t emins, const vec3_t emaxs, const mplane_t *p)
304 {
305         if (p->type < 3)
306                 return ((emaxs[p->type] >= p->dist) | ((emins[p->type] < p->dist) << 1));
307         switch(p->signbits)
308         {
309         default:
310         case 0:
311                 return (((p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2]) >= p->dist) | (((p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2]) < p->dist) << 1));
312         case 1:
313                 return (((p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2]) >= p->dist) | (((p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2]) < p->dist) << 1));
314         case 2:
315                 return (((p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2]) >= p->dist) | (((p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2]) < p->dist) << 1));
316         case 3:
317                 return (((p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2]) >= p->dist) | (((p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2]) < p->dist) << 1));
318         case 4:
319                 return (((p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2]) >= p->dist) | (((p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2]) < p->dist) << 1));
320         case 5:
321                 return (((p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2]) >= p->dist) | (((p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2]) < p->dist) << 1));
322         case 6:
323                 return (((p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2]) >= p->dist) | (((p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2]) < p->dist) << 1));
324         case 7:
325                 return (((p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2]) >= p->dist) | (((p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2]) < p->dist) << 1));
326         }
327 }
328
329 void AngleVectors (const vec3_t angles, vec3_t forward, vec3_t right, vec3_t up)
330 {
331         double angle, sr, sp, sy, cr, cp, cy;
332
333         angle = angles[YAW] * (M_PI*2 / 360);
334         sy = sin(angle);
335         cy = cos(angle);
336         angle = angles[PITCH] * (M_PI*2 / 360);
337         sp = sin(angle);
338         cp = cos(angle);
339         if (forward)
340         {
341                 forward[0] = cp*cy;
342                 forward[1] = cp*sy;
343                 forward[2] = -sp;
344         }
345         if (right || up)
346         {
347                 angle = angles[ROLL] * (M_PI*2 / 360);
348                 sr = sin(angle);
349                 cr = cos(angle);
350                 if (right)
351                 {
352                         right[0] = -1*(sr*sp*cy+cr*-sy);
353                         right[1] = -1*(sr*sp*sy+cr*cy);
354                         right[2] = -1*(sr*cp);
355                 }
356                 if (up)
357                 {
358                         up[0] = (cr*sp*cy+-sr*-sy);
359                         up[1] = (cr*sp*sy+-sr*cy);
360                         up[2] = cr*cp;
361                 }
362         }
363 }
364
365 void AngleVectorsFLU (const vec3_t angles, vec3_t forward, vec3_t left, vec3_t up)
366 {
367         double angle, sr, sp, sy, cr, cp, cy;
368
369         angle = angles[YAW] * (M_PI*2 / 360);
370         sy = sin(angle);
371         cy = cos(angle);
372         angle = angles[PITCH] * (M_PI*2 / 360);
373         sp = sin(angle);
374         cp = cos(angle);
375         if (forward)
376         {
377                 forward[0] = cp*cy;
378                 forward[1] = cp*sy;
379                 forward[2] = -sp;
380         }
381         if (left || up)
382         {
383                 angle = angles[ROLL] * (M_PI*2 / 360);
384                 sr = sin(angle);
385                 cr = cos(angle);
386                 if (left)
387                 {
388                         left[0] = sr*sp*cy+cr*-sy;
389                         left[1] = sr*sp*sy+cr*cy;
390                         left[2] = sr*cp;
391                 }
392                 if (up)
393                 {
394                         up[0] = cr*sp*cy+-sr*-sy;
395                         up[1] = cr*sp*sy+-sr*cy;
396                         up[2] = cr*cp;
397                 }
398         }
399 }
400
401 void AngleMatrix (const vec3_t angles, const vec3_t translate, vec_t matrix[][4])
402 {
403         double angle, sr, sp, sy, cr, cp, cy;
404
405         angle = angles[YAW] * (M_PI*2 / 360);
406         sy = sin(angle);
407         cy = cos(angle);
408         angle = angles[PITCH] * (M_PI*2 / 360);
409         sp = sin(angle);
410         cp = cos(angle);
411         angle = angles[ROLL] * (M_PI*2 / 360);
412         sr = sin(angle);
413         cr = cos(angle);
414         matrix[0][0] = cp*cy;
415         matrix[0][1] = sr*sp*cy+cr*-sy;
416         matrix[0][2] = cr*sp*cy+-sr*-sy;
417         matrix[0][3] = translate[0];
418         matrix[1][0] = cp*sy;
419         matrix[1][1] = sr*sp*sy+cr*cy;
420         matrix[1][2] = cr*sp*sy+-sr*cy;
421         matrix[1][3] = translate[1];
422         matrix[2][0] = -sp;
423         matrix[2][1] = sr*cp;
424         matrix[2][2] = cr*cp;
425         matrix[2][3] = translate[2];
426 }
427
428
429 // LordHavoc: renamed this to Length, and made the normal one a #define
430 float VectorNormalizeLength (vec3_t v)
431 {
432         float length, ilength;
433
434         length = v[0]*v[0] + v[1]*v[1] + v[2]*v[2];
435         length = sqrt (length);
436
437         if (length)
438         {
439                 ilength = 1/length;
440                 v[0] *= ilength;
441                 v[1] *= ilength;
442                 v[2] *= ilength;
443         }
444                 
445         return length;
446
447 }
448
449
450 /*
451 ================
452 R_ConcatRotations
453 ================
454 */
455 void R_ConcatRotations (const float in1[3*3], const float in2[3*3], float out[3*3])
456 {
457         out[0*3+0] = in1[0*3+0] * in2[0*3+0] + in1[0*3+1] * in2[1*3+0] + in1[0*3+2] * in2[2*3+0];
458         out[0*3+1] = in1[0*3+0] * in2[0*3+1] + in1[0*3+1] * in2[1*3+1] + in1[0*3+2] * in2[2*3+1];
459         out[0*3+2] = in1[0*3+0] * in2[0*3+2] + in1[0*3+1] * in2[1*3+2] + in1[0*3+2] * in2[2*3+2];
460         out[1*3+0] = in1[1*3+0] * in2[0*3+0] + in1[1*3+1] * in2[1*3+0] + in1[1*3+2] * in2[2*3+0];
461         out[1*3+1] = in1[1*3+0] * in2[0*3+1] + in1[1*3+1] * in2[1*3+1] + in1[1*3+2] * in2[2*3+1];
462         out[1*3+2] = in1[1*3+0] * in2[0*3+2] + in1[1*3+1] * in2[1*3+2] + in1[1*3+2] * in2[2*3+2];
463         out[2*3+0] = in1[2*3+0] * in2[0*3+0] + in1[2*3+1] * in2[1*3+0] + in1[2*3+2] * in2[2*3+0];
464         out[2*3+1] = in1[2*3+0] * in2[0*3+1] + in1[2*3+1] * in2[1*3+1] + in1[2*3+2] * in2[2*3+1];
465         out[2*3+2] = in1[2*3+0] * in2[0*3+2] + in1[2*3+1] * in2[1*3+2] + in1[2*3+2] * in2[2*3+2];
466 }
467
468
469 /*
470 ================
471 R_ConcatTransforms
472 ================
473 */
474 void R_ConcatTransforms (const float in1[3*4], const float in2[3*4], float out[3*4])
475 {
476         out[0*4+0] = in1[0*4+0] * in2[0*4+0] + in1[0*4+1] * in2[1*4+0] + in1[0*4+2] * in2[2*4+0];
477         out[0*4+1] = in1[0*4+0] * in2[0*4+1] + in1[0*4+1] * in2[1*4+1] + in1[0*4+2] * in2[2*4+1];
478         out[0*4+2] = in1[0*4+0] * in2[0*4+2] + in1[0*4+1] * in2[1*4+2] + in1[0*4+2] * in2[2*4+2];
479         out[0*4+3] = in1[0*4+0] * in2[0*4+3] + in1[0*4+1] * in2[1*4+3] + in1[0*4+2] * in2[2*4+3] + in1[0*4+3];
480         out[1*4+0] = in1[1*4+0] * in2[0*4+0] + in1[1*4+1] * in2[1*4+0] + in1[1*4+2] * in2[2*4+0];
481         out[1*4+1] = in1[1*4+0] * in2[0*4+1] + in1[1*4+1] * in2[1*4+1] + in1[1*4+2] * in2[2*4+1];
482         out[1*4+2] = in1[1*4+0] * in2[0*4+2] + in1[1*4+1] * in2[1*4+2] + in1[1*4+2] * in2[2*4+2];
483         out[1*4+3] = in1[1*4+0] * in2[0*4+3] + in1[1*4+1] * in2[1*4+3] + in1[1*4+2] * in2[2*4+3] + in1[1*4+3];
484         out[2*4+0] = in1[2*4+0] * in2[0*4+0] + in1[2*4+1] * in2[1*4+0] + in1[2*4+2] * in2[2*4+0];
485         out[2*4+1] = in1[2*4+0] * in2[0*4+1] + in1[2*4+1] * in2[1*4+1] + in1[2*4+2] * in2[2*4+1];
486         out[2*4+2] = in1[2*4+0] * in2[0*4+2] + in1[2*4+1] * in2[1*4+2] + in1[2*4+2] * in2[2*4+2];
487         out[2*4+3] = in1[2*4+0] * in2[0*4+3] + in1[2*4+1] * in2[1*4+3] + in1[2*4+2] * in2[2*4+3] + in1[2*4+3];
488 }
489
490
491 void Mathlib_Init(void)
492 {
493         int a;
494
495         // LordHavoc: setup 1.0f / N table for quick recipricols of integers
496         ixtable[0] = 0;
497         for (a = 1;a < 4096;a++)
498                 ixtable[a] = 1.0f / a;
499 }
500