// Copyright (c) 2010 The Chromium OS Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "matrixop.h" #include <math.h> #define PI 3.1415926535897932384626433832795f void Matrix4x4_Copy(Matrix4x4 dst, Matrix4x4 src) { int i; for (i = 0; i < 16; ++i) dst[i] = src[i]; } void Matrix4x4_Multiply(Matrix4x4 result, Matrix4x4 mat1, Matrix4x4 mat2) { Matrix4x4 tmp; int i, j, k; for (i = 0; i < 4; i++) { for (j = 0; j < 4; ++j) { tmp[i*4 + j] = 0; for (k = 0; k < 4; ++k) tmp[i*4 + j] += mat1[i*4 + k] * mat2[k*4 + j]; } } Matrix4x4_Copy(result, tmp); } void Matrix4x4_LoadIdentity(Matrix4x4 mat) { int i; for (i = 0; i < 16; ++i) mat[i] = 0; for (i = 0; i < 4; ++i) mat[i*4 + i] = 1.f; } void Matrix4x4_Scale(Matrix4x4 mat, float sx, float sy, float sz) { int i; for (i = 0; i < 4; ++i) { mat[0*4 + i] *= sx; mat[1*4 + i] *= sy; mat[2*4 + i] *= sz; } } void Matrix4x4_Translate(Matrix4x4 mat, float tx, float ty, float tz) { int i; for (i = 0; i < 4; ++i) mat[3*4 + i] += mat[0*4 + i] * tx + mat[1*4 + i] * ty + mat[2*4 + i] * tz; } static float normalize(float *ax, float *ay, float *az) { float norm = sqrtf((*ax) * (*ax) + (*ay) * (*ay) + (*az) * (*az)); if (norm > 0) { *ax /= norm; *ay /= norm; *az /= norm; } return norm; } void Matrix4x4_Rotate(Matrix4x4 mat, float angle, float ax, float ay, float az) { Matrix4x4 rot; float r = angle * PI / 180.f; float s = sinf(r); float c = cosf(r); float one_c = 1.f - c; float xx, yy, zz, xy, yz, xz, xs, ys, zs; float norm = normalize(&ax, &ay, &az); if (norm == 0 || angle == 0) return; xx = ax * ax; yy = ay * ay; zz = az * az; xy = ax * ay; yz = ay * az; xz = ax * az; xs = ax * s; ys = ay * s; zs = az * s; rot[0*4 + 0] = xx + (1.f - xx) * c; rot[1*4 + 0] = xy * one_c - zs; rot[2*4 + 0] = xz * one_c + ys; rot[3*4 + 0] = 0.f; rot[0*4 + 1] = xy * one_c + zs; rot[1*4 + 1] = yy + (1.f - yy) * c; rot[2*4 + 1] = yz * one_c - xs; rot[3*4 + 1] = 0.f; rot[0*4 + 2] = xz * one_c - ys; rot[1*4 + 2] = yz * one_c + xs; rot[2*4 + 2] = zz + (1.f - zz) * c; rot[3*4 + 2] = 0.f; rot[0*4 + 3] = 0.f; rot[1*4 + 3] = 0.f; rot[2*4 + 3] = 0.f; rot[3*4 + 3] = 1.f; Matrix4x4_Multiply(mat, rot, mat); } void Matrix4x4_Frustum(Matrix4x4 mat, float left, float right, float bottom, float top, float near, float far) { float dx = right - left; float dy = top - bottom; float dz = far - near; Matrix4x4 frust; if (near <= 0.f || far <= 0.f || dx <= 0.f || dy <= 0.f || dz <= 0.f) return; frust[0*4 + 0] = 2.f * near / dx; frust[0*4 + 1] = 0.f; frust[0*4 + 2] = 0.f; frust[0*4 + 3] = 0.f; frust[1*4 + 0] = 0.f; frust[1*4 + 1] = 2.f * near / dy; frust[1*4 + 2] = 0.f; frust[1*4 + 3] = 0.f; frust[2*4 + 0] = (right + left) / dx; frust[2*4 + 1] = (top + bottom) / dy; frust[2*4 + 2] = -(near + far) / dz; frust[2*4 + 3] = -1.f; frust[3*4 + 0] = 0.f; frust[3*4 + 1] = 0.f; frust[3*4 + 2] = -2.f * near * far / dz; frust[3*4 + 3] = 0.f; Matrix4x4_Multiply(mat, frust, mat); } void Matrix4x4_Perspective(Matrix4x4 mat, float fovy, float aspect, float nearZ, float farZ) { float frustumW, frustumH; frustumH = tanf(fovy / 360.f * PI) * nearZ; frustumW = frustumH * aspect; Matrix4x4_Frustum(mat, -frustumW, frustumW, -frustumH, frustumH, nearZ, farZ); } void Matrix4x4_Transform(Matrix4x4 mat, float *x, float *y, float *z) { float tx = mat[0*4 + 0] * (*x) + mat[1*4 + 0] * (*y) + mat[2*4 + 0] * (*z); float ty = mat[0*4 + 1] * (*x) + mat[1*4 + 1] * (*y) + mat[2*4 + 1] * (*z); float tz = mat[0*4 + 2] * (*x) + mat[1*4 + 2] * (*y) + mat[2*4 + 2] * (*z); *x = tx; *y = ty; *z = tz; }