conworlds

changeset 13:283cdfa7dda2

added a crapload of code from goat3dgfx
author John Tsiombikas <nuclear@member.fsf.org>
date Sun, 24 Aug 2014 09:41:24 +0300 (2014-08-24)
parents 778ed91cb7fd
children 423d4e6728cb
files Makefile conworlds.sln conworlds.vcxproj conworlds.vcxproj.filters src/assload.cc src/assload.h src/dataset.h src/dataset.inl src/game.cc src/gameopt.cc src/gameopt.h src/geom.cc src/geom.h src/light.cc src/light.h src/logger.cc src/logger.h src/main.cc src/material.cc src/material.h src/mesh.cc src/mesh.h src/object.cc src/object.h src/scene.cc src/scene.h src/shader.cc src/shader.h src/texture.cc src/texture.h src/unistate.cc src/unistate.h src/xform_node.cc src/xform_node.h vrchess.sln vrchess.vcxproj vrchess.vcxproj.filters
diffstat 37 files changed, 5398 insertions(+), 274 deletions(-) [+]
line diff
     1.1 --- a/Makefile	Sat Aug 23 12:03:29 2014 +0300
     1.2 +++ b/Makefile	Sun Aug 24 09:41:24 2014 +0300
     1.3 @@ -2,7 +2,7 @@
     1.4  ccsrc = $(wildcard src/*.cc)
     1.5  obj = $(csrc:.c=.o) $(ccsrc:.cc=.o)
     1.6  dep = $(obj:.o=.d)
     1.7 -bin = vrchess
     1.8 +bin = conworlds
     1.9  
    1.10  # comment out to disable LibOVR (oculus sdk)
    1.11  #ovr_cflags = -DUSE_LIBOVR
    1.12 @@ -13,7 +13,7 @@
    1.13  
    1.14  CFLAGS = -pedantic -Wall -g $(ovr_cflags)
    1.15  CXXFLAGS = -std=c++11 $(CFLAGS)
    1.16 -LDFLAGS = $(libgl_$(sys)) -lm -lvmath -limago $(ovr_libs)
    1.17 +LDFLAGS = $(libgl_$(sys)) -lm -lvmath -limago -lanim $(ovr_libs)
    1.18  
    1.19  libgl_unix = -lGL -lGLU -lglut -lGLEW
    1.20  libgl_mac = -framework OpenGL -framework GLUT -lGLEW
     2.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     2.2 +++ b/conworlds.sln	Sun Aug 24 09:41:24 2014 +0300
     2.3 @@ -0,0 +1,20 @@
     2.4 +
     2.5 +Microsoft Visual Studio Solution File, Format Version 12.00
     2.6 +# Visual Studio 2012
     2.7 +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "conworlds", "conworlds.vcxproj", "{714906B6-FD4C-4ECC-990C-247FA46B8801}"
     2.8 +EndProject
     2.9 +Global
    2.10 +	GlobalSection(SolutionConfigurationPlatforms) = preSolution
    2.11 +		Debug|Win32 = Debug|Win32
    2.12 +		Release|Win32 = Release|Win32
    2.13 +	EndGlobalSection
    2.14 +	GlobalSection(ProjectConfigurationPlatforms) = postSolution
    2.15 +		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Debug|Win32.ActiveCfg = Debug|Win32
    2.16 +		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Debug|Win32.Build.0 = Debug|Win32
    2.17 +		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Release|Win32.ActiveCfg = Release|Win32
    2.18 +		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Release|Win32.Build.0 = Release|Win32
    2.19 +	EndGlobalSection
    2.20 +	GlobalSection(SolutionProperties) = preSolution
    2.21 +		HideSolutionNode = FALSE
    2.22 +	EndGlobalSection
    2.23 +EndGlobal
     3.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     3.2 +++ b/conworlds.vcxproj	Sun Aug 24 09:41:24 2014 +0300
     3.3 @@ -0,0 +1,117 @@
     3.4 +<?xml version="1.0" encoding="utf-8"?>
     3.5 +<Project DefaultTargets="Build" ToolsVersion="12.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
     3.6 +  <ItemGroup Label="ProjectConfigurations">
     3.7 +    <ProjectConfiguration Include="Debug|Win32">
     3.8 +      <Configuration>Debug</Configuration>
     3.9 +      <Platform>Win32</Platform>
    3.10 +    </ProjectConfiguration>
    3.11 +    <ProjectConfiguration Include="Release|Win32">
    3.12 +      <Configuration>Release</Configuration>
    3.13 +      <Platform>Win32</Platform>
    3.14 +    </ProjectConfiguration>
    3.15 +  </ItemGroup>
    3.16 +  <PropertyGroup Label="Globals">
    3.17 +    <ProjectGuid>{714906B6-FD4C-4ECC-990C-247FA46B8801}</ProjectGuid>
    3.18 +    <Keyword>Win32Proj</Keyword>
    3.19 +    <RootNamespace>conworlds</RootNamespace>
    3.20 +  </PropertyGroup>
    3.21 +  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
    3.22 +  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
    3.23 +    <ConfigurationType>Application</ConfigurationType>
    3.24 +    <UseDebugLibraries>true</UseDebugLibraries>
    3.25 +    <PlatformToolset>v120</PlatformToolset>
    3.26 +    <CharacterSet>MultiByte</CharacterSet>
    3.27 +  </PropertyGroup>
    3.28 +  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
    3.29 +    <ConfigurationType>Application</ConfigurationType>
    3.30 +    <UseDebugLibraries>false</UseDebugLibraries>
    3.31 +    <PlatformToolset>v120</PlatformToolset>
    3.32 +    <WholeProgramOptimization>false</WholeProgramOptimization>
    3.33 +    <CharacterSet>MultiByte</CharacterSet>
    3.34 +  </PropertyGroup>
    3.35 +  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
    3.36 +  <ImportGroup Label="ExtensionSettings">
    3.37 +  </ImportGroup>
    3.38 +  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
    3.39 +    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
    3.40 +  </ImportGroup>
    3.41 +  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
    3.42 +    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
    3.43 +  </ImportGroup>
    3.44 +  <PropertyGroup Label="UserMacros" />
    3.45 +  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
    3.46 +    <LinkIncremental>true</LinkIncremental>
    3.47 +  </PropertyGroup>
    3.48 +  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
    3.49 +    <LinkIncremental>false</LinkIncremental>
    3.50 +  </PropertyGroup>
    3.51 +  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
    3.52 +    <ClCompile>
    3.53 +      <PrecompiledHeader>
    3.54 +      </PrecompiledHeader>
    3.55 +      <WarningLevel>Level3</WarningLevel>
    3.56 +      <Optimization>Disabled</Optimization>
    3.57 +      <PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions);USE_LIBOVR</PreprocessorDefinitions>
    3.58 +      <DisableSpecificWarnings>4996;4244;4305</DisableSpecificWarnings>
    3.59 +      <AdditionalIncludeDirectories>$(SolutionDir)\src</AdditionalIncludeDirectories>
    3.60 +    </ClCompile>
    3.61 +    <Link>
    3.62 +      <SubSystem>Console</SubSystem>
    3.63 +      <GenerateDebugInformation>true</GenerateDebugInformation>
    3.64 +      <AdditionalDependencies>opengl32.lib;freeglutd.lib;glew32.lib;libvmath.lib;libimago2.lib;jpeglib.lib;libpng.lib;zlib.lib;libovrd.lib;ws2_32.lib;%(AdditionalDependencies)</AdditionalDependencies>
    3.65 +    </Link>
    3.66 +  </ItemDefinitionGroup>
    3.67 +  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
    3.68 +    <ClCompile>
    3.69 +      <WarningLevel>Level3</WarningLevel>
    3.70 +      <PrecompiledHeader>
    3.71 +      </PrecompiledHeader>
    3.72 +      <Optimization>MaxSpeed</Optimization>
    3.73 +      <FunctionLevelLinking>true</FunctionLevelLinking>
    3.74 +      <IntrinsicFunctions>true</IntrinsicFunctions>
    3.75 +      <PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions);USE_LIBOVR</PreprocessorDefinitions>
    3.76 +      <DisableSpecificWarnings>4996;4244;4305</DisableSpecificWarnings>
    3.77 +      <AdditionalIncludeDirectories>$(SolutionDir)\src</AdditionalIncludeDirectories>
    3.78 +    </ClCompile>
    3.79 +    <Link>
    3.80 +      <SubSystem>Console</SubSystem>
    3.81 +      <GenerateDebugInformation>true</GenerateDebugInformation>
    3.82 +      <EnableCOMDATFolding>true</EnableCOMDATFolding>
    3.83 +      <OptimizeReferences>true</OptimizeReferences>
    3.84 +      <AdditionalDependencies>opengl32.lib;freeglut.lib;glew32.lib;libvmath.lib;libimago2.lib;jpeglib.lib;libpng.lib;zlib.lib;libovr.lib;ws2_32.lib;%(AdditionalDependencies)</AdditionalDependencies>
    3.85 +    </Link>
    3.86 +  </ItemDefinitionGroup>
    3.87 +  <ItemGroup>
    3.88 +    <ClCompile Include="src\camera.cc" />
    3.89 +    <ClCompile Include="src\game.cc" />
    3.90 +    <ClCompile Include="src\image.cc" />
    3.91 +    <ClCompile Include="src\main.cc" />
    3.92 +    <ClCompile Include="src\opengl.cc" />
    3.93 +    <ClCompile Include="src\sdr.c" />
    3.94 +    <ClCompile Include="src\texture.cc" />
    3.95 +    <ClCompile Include="src\vr\mathutil.c" />
    3.96 +    <ClCompile Include="src\vr\opt.c" />
    3.97 +    <ClCompile Include="src\vr\rbtree.c" />
    3.98 +    <ClCompile Include="src\vr\vr.c" />
    3.99 +    <ClCompile Include="src\vr\vr_libovr.c" />
   3.100 +    <ClCompile Include="src\vr\vr_modules.c" />
   3.101 +    <ClCompile Include="src\vr\vr_null.c" />
   3.102 +    <ClCompile Include="src\vr\vr_openhmd.c" />
   3.103 +  </ItemGroup>
   3.104 +  <ItemGroup>
   3.105 +    <ClInclude Include="src\camera.h" />
   3.106 +    <ClInclude Include="src\game.h" />
   3.107 +    <ClInclude Include="src\image.h" />
   3.108 +    <ClInclude Include="src\opengl.h" />
   3.109 +    <ClInclude Include="src\sdr.h" />
   3.110 +    <ClInclude Include="src\texture.h" />
   3.111 +    <ClInclude Include="src\vr\mathutil.h" />
   3.112 +    <ClInclude Include="src\vr\opt.h" />
   3.113 +    <ClInclude Include="src\vr\rbtree.h" />
   3.114 +    <ClInclude Include="src\vr\vr.h" />
   3.115 +    <ClInclude Include="src\vr\vr_impl.h" />
   3.116 +  </ItemGroup>
   3.117 +  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
   3.118 +  <ImportGroup Label="ExtensionTargets">
   3.119 +  </ImportGroup>
   3.120 +</Project>
     4.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     4.2 +++ b/conworlds.vcxproj.filters	Sun Aug 24 09:41:24 2014 +0300
     4.3 @@ -0,0 +1,94 @@
     4.4 +<?xml version="1.0" encoding="utf-8"?>
     4.5 +<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
     4.6 +  <ItemGroup>
     4.7 +    <Filter Include="src">
     4.8 +      <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>
     4.9 +      <Extensions>cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx;h;inl</Extensions>
    4.10 +    </Filter>
    4.11 +    <Filter Include="src\vr">
    4.12 +      <UniqueIdentifier>{e881ab02-1a45-43f6-a15d-ee7f77256a1e}</UniqueIdentifier>
    4.13 +    </Filter>
    4.14 +  </ItemGroup>
    4.15 +  <ItemGroup>
    4.16 +    <ClCompile Include="src\main.cc">
    4.17 +      <Filter>src</Filter>
    4.18 +    </ClCompile>
    4.19 +    <ClCompile Include="src\camera.cc">
    4.20 +      <Filter>src</Filter>
    4.21 +    </ClCompile>
    4.22 +    <ClCompile Include="src\opengl.cc">
    4.23 +      <Filter>src</Filter>
    4.24 +    </ClCompile>
    4.25 +    <ClCompile Include="src\sdr.c">
    4.26 +      <Filter>src</Filter>
    4.27 +    </ClCompile>
    4.28 +    <ClCompile Include="src\game.cc">
    4.29 +      <Filter>src</Filter>
    4.30 +    </ClCompile>
    4.31 +    <ClCompile Include="src\image.cc">
    4.32 +      <Filter>src</Filter>
    4.33 +    </ClCompile>
    4.34 +    <ClCompile Include="src\texture.cc">
    4.35 +      <Filter>src</Filter>
    4.36 +    </ClCompile>
    4.37 +    <ClCompile Include="src\vr\vr.c">
    4.38 +      <Filter>src\vr</Filter>
    4.39 +    </ClCompile>
    4.40 +    <ClCompile Include="src\vr\vr_libovr.c">
    4.41 +      <Filter>src\vr</Filter>
    4.42 +    </ClCompile>
    4.43 +    <ClCompile Include="src\vr\vr_modules.c">
    4.44 +      <Filter>src\vr</Filter>
    4.45 +    </ClCompile>
    4.46 +    <ClCompile Include="src\vr\vr_null.c">
    4.47 +      <Filter>src\vr</Filter>
    4.48 +    </ClCompile>
    4.49 +    <ClCompile Include="src\vr\rbtree.c">
    4.50 +      <Filter>src\vr</Filter>
    4.51 +    </ClCompile>
    4.52 +    <ClCompile Include="src\vr\opt.c">
    4.53 +      <Filter>src\vr</Filter>
    4.54 +    </ClCompile>
    4.55 +    <ClCompile Include="src\vr\vr_openhmd.c">
    4.56 +      <Filter>src\vr</Filter>
    4.57 +    </ClCompile>
    4.58 +    <ClCompile Include="src\vr\mathutil.c">
    4.59 +      <Filter>src\vr</Filter>
    4.60 +    </ClCompile>
    4.61 +  </ItemGroup>
    4.62 +  <ItemGroup>
    4.63 +    <ClInclude Include="src\camera.h">
    4.64 +      <Filter>src</Filter>
    4.65 +    </ClInclude>
    4.66 +    <ClInclude Include="src\game.h">
    4.67 +      <Filter>src</Filter>
    4.68 +    </ClInclude>
    4.69 +    <ClInclude Include="src\image.h">
    4.70 +      <Filter>src</Filter>
    4.71 +    </ClInclude>
    4.72 +    <ClInclude Include="src\opengl.h">
    4.73 +      <Filter>src</Filter>
    4.74 +    </ClInclude>
    4.75 +    <ClInclude Include="src\sdr.h">
    4.76 +      <Filter>src</Filter>
    4.77 +    </ClInclude>
    4.78 +    <ClInclude Include="src\texture.h">
    4.79 +      <Filter>src</Filter>
    4.80 +    </ClInclude>
    4.81 +    <ClInclude Include="src\vr\vr.h">
    4.82 +      <Filter>src\vr</Filter>
    4.83 +    </ClInclude>
    4.84 +    <ClInclude Include="src\vr\vr_impl.h">
    4.85 +      <Filter>src\vr</Filter>
    4.86 +    </ClInclude>
    4.87 +    <ClInclude Include="src\vr\rbtree.h">
    4.88 +      <Filter>src\vr</Filter>
    4.89 +    </ClInclude>
    4.90 +    <ClInclude Include="src\vr\opt.h">
    4.91 +      <Filter>src\vr</Filter>
    4.92 +    </ClInclude>
    4.93 +    <ClInclude Include="src\vr\mathutil.h">
    4.94 +      <Filter>src\vr</Filter>
    4.95 +    </ClInclude>
    4.96 +  </ItemGroup>
    4.97 +</Project>
    4.98 \ No newline at end of file
     5.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     5.2 +++ b/src/assload.cc	Sun Aug 24 09:41:24 2014 +0300
     5.3 @@ -0,0 +1,437 @@
     5.4 +#include <stdio.h>
     5.5 +#include "assload.h"
     5.6 +#include "logger.h"
     5.7 +
     5.8 +#ifdef USE_ASSIMP
     5.9 +
    5.10 +#include <vector>
    5.11 +#include <map>
    5.12 +#include "assimp/cimport.h"
    5.13 +#include "assimp/scene.h"
    5.14 +#include "assimp/postprocess.h"
    5.15 +#include "texman.h"
    5.16 +#include "material.h"
    5.17 +#include "scene.h"
    5.18 +
    5.19 +using namespace std;
    5.20 +
    5.21 +static bool load_material(Material *mat, const aiMaterial *aimat);
    5.22 +static Object *load_node(const aiScene *aiscn, const aiNode *ainode);
    5.23 +static Mesh *load_mesh(const aiScene *aiscn, const aiMesh *aimesh);
    5.24 +static Curve *load_curve(const aiScene *aiscn, const aiMesh *aimesh);
    5.25 +static bool load_bones(Mesh *mesh, const aiMesh *aimesh);
    5.26 +
    5.27 +static Vector3 assimp_vector(const aiVector3D &v);
    5.28 +static Quaternion assimp_quat(const aiQuaternion &q);
    5.29 +static Matrix4x4 assimp_matrix(const aiMatrix4x4 &aim);
    5.30 +static long assimp_time(const aiAnimation *anim, double aitime);
    5.31 +static void print_hierarchy(const aiNode *node);
    5.32 +
    5.33 +static map<string, Object*> obj_by_name;
    5.34 +static map<aiMesh*, Mesh*> mesh_by_aimesh;
    5.35 +
    5.36 +bool load_ass(Scene *scn, const char *fname)
    5.37 +{
    5.38 +	static bool init_done;
    5.39 +
    5.40 +	if(!init_done) {
    5.41 +		static aiLogStream log_stream = aiGetPredefinedLogStream(aiDefaultLogStream_STDOUT, 0);
    5.42 +		aiAttachLogStream(&log_stream);
    5.43 +		//aiEnableVerboseLogging(1);
    5.44 +		init_done = true;
    5.45 +	}
    5.46 +
    5.47 +	unsigned int proc_flags = aiProcess_JoinIdenticalVertices |
    5.48 +		aiProcess_CalcTangentSpace |
    5.49 +		aiProcess_Triangulate |
    5.50 +		aiProcess_SortByPType |
    5.51 +		aiProcess_FlipUVs;
    5.52 +
    5.53 +	const aiScene *aiscn = aiImportFile(datafile_path(fname).c_str(), proc_flags);
    5.54 +	if(!aiscn) {
    5.55 +		error_log("failed to load file: %s\n", fname);
    5.56 +		return false;
    5.57 +	}
    5.58 +
    5.59 +	info_log("NODE HIERARCHY:\n");
    5.60 +	print_hierarchy(aiscn->mRootNode);
    5.61 +	info_log("-------------------\n");
    5.62 +
    5.63 +	Vector3 root_pos, root_scaling(1.0, 1.0, 1.0);
    5.64 +	Quaternion root_rot;
    5.65 +
    5.66 +	if(aiscn->mRootNode) {
    5.67 +		Matrix4x4 root_matrix = assimp_matrix(aiscn->mRootNode->mTransformation);
    5.68 +		root_pos = root_matrix.get_translation();
    5.69 +		root_rot = root_matrix.get_rotation_quat();
    5.70 +		root_scaling = root_matrix.get_scaling();
    5.71 +	}
    5.72 +
    5.73 +	// load all meshes
    5.74 +	for(unsigned int i=0; i<aiscn->mNumMeshes; i++) {
    5.75 +		aiMesh *aimesh = aiscn->mMeshes[i];
    5.76 +		Mesh *mesh;
    5.77 +		Curve *curve;
    5.78 +
    5.79 +		switch(aimesh->mPrimitiveTypes) {
    5.80 +		case aiPrimitiveType_TRIANGLE:
    5.81 +			if((mesh = load_mesh(aiscn, aimesh))) {
    5.82 +				mesh_by_aimesh[aimesh] = mesh;
    5.83 +				scn->meshes.push_back(mesh);
    5.84 +			}
    5.85 +			break;
    5.86 +
    5.87 +		case aiPrimitiveType_LINE:
    5.88 +			if((curve = load_curve(aiscn, aimesh))) {
    5.89 +				scn->curves.push_back(curve);
    5.90 +			}
    5.91 +			break;
    5.92 +
    5.93 +		default:
    5.94 +			error_log("unsupported primitive type: %u\n", aimesh->mPrimitiveTypes);
    5.95 +			break;
    5.96 +		}
    5.97 +	}
    5.98 +
    5.99 +	// load all the nodes recursively
   5.100 +	for(unsigned int i=0; i<aiscn->mRootNode->mNumChildren; i++) {
   5.101 +		Object *obj = load_node(aiscn, aiscn->mRootNode->mChildren[i]);
   5.102 +		if(obj) {
   5.103 +			Object *dummy = new Object;
   5.104 +			dummy->set_name((string("dummyroot_") + string(obj->get_name())).c_str());
   5.105 +			dummy->set_position(root_pos);
   5.106 +			dummy->set_rotation(root_rot);
   5.107 +			dummy->set_scaling(root_scaling);
   5.108 +			dummy->add_child(obj);
   5.109 +
   5.110 +			obj = dummy;
   5.111 +			scn->objects.push_back(obj);
   5.112 +		}
   5.113 +	}
   5.114 +
   5.115 +	// load and attach the bones to the meshes
   5.116 +	for(unsigned int i=0; i<aiscn->mNumMeshes; i++) {
   5.117 +		aiMesh *aimesh = aiscn->mMeshes[i];
   5.118 +
   5.119 +		Mesh *mesh = mesh_by_aimesh[aimesh];
   5.120 +		load_bones(mesh, aimesh);
   5.121 +	}
   5.122 +
   5.123 +	obj_by_name.clear();
   5.124 +	mesh_by_aimesh.clear();
   5.125 +
   5.126 +	aiReleaseImport(aiscn);
   5.127 +	return true;
   5.128 +}
   5.129 +
   5.130 +static bool load_material(Material *mat, const aiMaterial *aimat)
   5.131 +{
   5.132 +	aiColor4D aicol;
   5.133 +	float shin, shin_str;
   5.134 +
   5.135 +	if(aiGetMaterialColor(aimat, AI_MATKEY_COLOR_DIFFUSE, &aicol) == 0) {
   5.136 +		mat->diffuse = Vector3(aicol[0], aicol[1], aicol[2]);
   5.137 +	}
   5.138 +	if(aiGetMaterialColor(aimat, AI_MATKEY_COLOR_SPECULAR, &aicol) == 0) {
   5.139 +		mat->specular = Vector3(aicol[0], aicol[1], aicol[2]);
   5.140 +	}
   5.141 +
   5.142 +	unsigned int count = 1;
   5.143 +	if(aiGetMaterialFloatArray(aimat, AI_MATKEY_SHININESS_STRENGTH, &shin_str, &count) != 0) {
   5.144 +		shin_str = 1.0;
   5.145 +	}
   5.146 +	if(aiGetMaterialFloatArray(aimat, AI_MATKEY_SHININESS, &shin, &count) == 0) {
   5.147 +		// XXX can't remember how I came up with this...
   5.148 +		mat->shininess = shin * shin_str * 0.0001 * 128.0;
   5.149 +	}
   5.150 +
   5.151 +	// load textures
   5.152 +	struct { int type; aiTextureType aitype; } textypes[] = {
   5.153 +		{TEX_DIFFUSE, aiTextureType_DIFFUSE},
   5.154 +		{TEX_NORMAL, aiTextureType_NORMALS},
   5.155 +		{TEX_SPECULAR, aiTextureType_SPECULAR}
   5.156 +	};
   5.157 +
   5.158 +	for(int i=0; i<sizeof textypes / sizeof *textypes; i++) {
   5.159 +		aiString aipath;
   5.160 +
   5.161 +		if(aiGetMaterialTexture(aimat, textypes[i].aitype, 0, &aipath) == 0) {
   5.162 +			char *tmp, *fname = aipath.data;
   5.163 +
   5.164 +			if((tmp = strrchr(fname, '/'))) {
   5.165 +				fname = tmp + 1;
   5.166 +			}
   5.167 +			if((tmp = strrchr(fname, '\\'))) {
   5.168 +				fname = tmp + 1;
   5.169 +			}
   5.170 +
   5.171 +			if(*fname) {
   5.172 +				mat->tex[textypes[i].type] = texset.get(fname);
   5.173 +			}
   5.174 +		}
   5.175 +	}
   5.176 +
   5.177 +	return true;
   5.178 +}
   5.179 +
   5.180 +static Object *load_node(const aiScene *aiscn, const aiNode *ainode)
   5.181 +{
   5.182 +	Object *obj = new Object;
   5.183 +	obj->set_name(ainode->mName.data);
   5.184 +
   5.185 +	if(ainode->mNumMeshes) {
   5.186 +		if(ainode->mNumMeshes > 1) {
   5.187 +			info_log("%s warning: node %s has more than one meshes (%u)\n", __FUNCTION__,
   5.188 +					ainode->mName.data, ainode->mNumMeshes);
   5.189 +		}
   5.190 +
   5.191 +		aiMesh *aimesh = aiscn->mMeshes[ainode->mMeshes[0]];
   5.192 +		obj->set_mesh(mesh_by_aimesh[aimesh]);
   5.193 +
   5.194 +		// also grab the material of this mesh
   5.195 +		load_material(&obj->material, aiscn->mMaterials[aimesh->mMaterialIndex]);
   5.196 +	}
   5.197 +
   5.198 +	// if there are animations, grab the first and try to use it
   5.199 +	if(aiscn->mNumAnimations) {
   5.200 +		aiAnimation *aianim = aiscn->mAnimations[0];
   5.201 +		aiNodeAnim *ainodeanim = 0;
   5.202 +		for(unsigned int i=0; i<aianim->mNumChannels; i++) {
   5.203 +			if(strcmp(aianim->mChannels[i]->mNodeName.data, ainode->mName.data) == 0) {
   5.204 +				ainodeanim = aianim->mChannels[i];
   5.205 +				break;
   5.206 +			}
   5.207 +		}
   5.208 +
   5.209 +		if(ainodeanim) {
   5.210 +			// load all position (translation) keyframes
   5.211 +			for(unsigned int i=0; i<ainodeanim->mNumPositionKeys; i++) {
   5.212 +				Vector3 pos = assimp_vector(ainodeanim->mPositionKeys[i].mValue);
   5.213 +				long msec = assimp_time(aianim, ainodeanim->mPositionKeys[i].mTime);
   5.214 +				obj->set_position(pos, msec);
   5.215 +			}
   5.216 +
   5.217 +			// load all rotation keyframes
   5.218 +			for(unsigned int i=0; i<ainodeanim->mNumRotationKeys; i++) {
   5.219 +				Quaternion rot = assimp_quat(ainodeanim->mRotationKeys[i].mValue);
   5.220 +				if(rot.length_sq() < SMALL_NUMBER) {
   5.221 +					continue;
   5.222 +				}
   5.223 +				rot.normalize();
   5.224 +				long msec = assimp_time(aianim, ainodeanim->mRotationKeys[i].mTime);
   5.225 +				obj->set_rotation(rot, msec);
   5.226 +			}
   5.227 +
   5.228 +			// load all scaling keyframes
   5.229 +			for(unsigned int i=0; i<ainodeanim->mNumScalingKeys; i++) {
   5.230 +				Vector3 scale = assimp_vector(ainodeanim->mScalingKeys[i].mValue);
   5.231 +				long msec = assimp_time(aianim, ainodeanim->mScalingKeys[i].mTime);
   5.232 +				obj->set_scaling(scale, msec);
   5.233 +			}
   5.234 +
   5.235 +			obj->set_extrapolator(EXTRAP_REPEAT);	// loop animation
   5.236 +		} else {
   5.237 +			Matrix4x4 local_matrix = assimp_matrix(ainode->mTransformation);
   5.238 +			obj->set_local_matrix(local_matrix);
   5.239 +		}
   5.240 +	}
   5.241 +
   5.242 +	/* recurse to all children */
   5.243 +	for(unsigned int i=0; i<ainode->mNumChildren; i++) {
   5.244 +		Object *child = load_node(aiscn, ainode->mChildren[i]);
   5.245 +		if(child) {
   5.246 +			obj->add_child(child);
   5.247 +		}
   5.248 +	}
   5.249 +
   5.250 +	obj_by_name[obj->get_name()] = obj;
   5.251 +	return obj;
   5.252 +}
   5.253 +
   5.254 +static Mesh *load_mesh(const aiScene *aiscn, const aiMesh *aimesh)
   5.255 +{
   5.256 +	Mesh *mesh = new Mesh;
   5.257 +
   5.258 +	int num_verts = aimesh->mNumVertices;
   5.259 +	int num_faces = aimesh->mNumFaces;
   5.260 +
   5.261 +	mesh->set_attrib_data(MESH_ATTR_VERTEX, 3, num_verts, (float*)aimesh->mVertices);
   5.262 +
   5.263 +	if(aimesh->mNormals) {
   5.264 +		mesh->set_attrib_data(MESH_ATTR_NORMAL, 3, num_verts, (float*)aimesh->mNormals);
   5.265 +	}
   5.266 +	if(aimesh->mTangents) {
   5.267 +		mesh->set_attrib_data(MESH_ATTR_TANGENT, 3, num_verts, (float*)aimesh->mTangents);
   5.268 +	}
   5.269 +	if(aimesh->mTextureCoords[0]) {
   5.270 +		mesh->set_attrib_data(MESH_ATTR_TEXCOORD, 3, num_verts, (float*)aimesh->mTextureCoords[0]);
   5.271 +	}
   5.272 +
   5.273 +	if(aimesh->mBones) {
   5.274 +		float *weights = mesh->set_attrib_data(MESH_ATTR_BONEWEIGHTS, 4, num_verts, 0);
   5.275 +		float *boneidx = mesh->set_attrib_data(MESH_ATTR_BONEIDX, 4, num_verts, 0);
   5.276 +
   5.277 +		memset(weights, 0, num_verts * 4 * sizeof *weights);
   5.278 +		memset(boneidx, 0, num_verts * 4 * sizeof *boneidx);
   5.279 +
   5.280 +		int *vertex_bone_count = new int[num_verts];
   5.281 +		memset(vertex_bone_count, 0, num_verts * sizeof *vertex_bone_count);
   5.282 +
   5.283 +		for(unsigned int i=0; i<aimesh->mNumBones; i++) {
   5.284 +			aiBone *aibone = aimesh->mBones[i];
   5.285 +
   5.286 +			// for every vertex affected by this bone:
   5.287 +			for(unsigned int j=0; j<aibone->mNumWeights; j++) {
   5.288 +				aiVertexWeight *aiweight = aibone->mWeights + j;
   5.289 +				int vidx = aiweight->mVertexId;
   5.290 +				int vert_boneidx = vertex_bone_count[vidx];
   5.291 +				if(vert_boneidx >= 4) {
   5.292 +					error_log("WARNING vertex with more than 4 bones found\n");
   5.293 +					continue;
   5.294 +				}
   5.295 +
   5.296 +				weights[vidx * 4 + vert_boneidx] = aiweight->mWeight;
   5.297 +				boneidx[vidx * 4 + vert_boneidx] = (float)i;
   5.298 +				vertex_bone_count[vidx]++;
   5.299 +			}
   5.300 +		}
   5.301 +
   5.302 +		delete [] vertex_bone_count;
   5.303 +
   5.304 +		// normalize weights
   5.305 +		for(int i=0; i<num_verts; i++) {
   5.306 +
   5.307 +			float wsum = 0.0f;
   5.308 +
   5.309 +			for(int j=0; j<4; j++) {
   5.310 +				wsum += weights[i * 4 + j];
   5.311 +			}
   5.312 +
   5.313 +			if(1.0 - wsum > 1e-4) {
   5.314 +				error_log("WARNING vertex with weights < 1 (%f), normalizing...\n", wsum);
   5.315 +
   5.316 +				if(wsum < 1e-6) {
   5.317 +					// this is clearly broken, let's use the first bone in full
   5.318 +					weights[i * 4] = 1.0;
   5.319 +				} else {
   5.320 +					weights[i * 4] /= wsum;
   5.321 +					weights[i * 4 + 1] /= wsum;
   5.322 +					weights[i * 4 + 2] /= wsum;
   5.323 +					weights[i * 4 + 3] /= wsum;
   5.324 +				}
   5.325 +			}
   5.326 +		}
   5.327 +	}
   5.328 +
   5.329 +	unsigned int *iptr = mesh->set_index_data(num_faces * 3);
   5.330 +	for(int i=0; i<num_faces; i++) {
   5.331 +		for(int j=0; j<3; j++) {
   5.332 +			*iptr++ = aimesh->mFaces[i].mIndices[j];
   5.333 +		}
   5.334 +	}
   5.335 +
   5.336 +	return mesh;
   5.337 +}
   5.338 +
   5.339 +static Curve *load_curve(const aiScene *aiscn, const aiMesh *aimesh)
   5.340 +{
   5.341 +	Curve *curve = new Curve;
   5.342 +
   5.343 +	for(unsigned int i=0; i<aimesh->mNumVertices; i++) {
   5.344 +		Vector3 pt = assimp_vector(aimesh->mVertices[i]);
   5.345 +		curve->add_point(pt);
   5.346 +	}
   5.347 +	info_log("loaded curve with %d points\n", aimesh->mNumVertices);
   5.348 +
   5.349 +	return curve;
   5.350 +}
   5.351 +
   5.352 +static bool load_bones(Mesh *mesh, const aiMesh *aimesh)
   5.353 +{
   5.354 +	if(!aimesh->mNumBones) {
   5.355 +		return false;
   5.356 +	}
   5.357 +
   5.358 +	for(unsigned int i=0; i<aimesh->mNumBones; i++) {
   5.359 +		aiBone *aibone = aimesh->mBones[i];
   5.360 +		Object *obj = obj_by_name[aibone->mName.data];
   5.361 +		if(!obj) {
   5.362 +			error_log("bone %s not found\n", aibone->mName.data);
   5.363 +			continue;
   5.364 +		}
   5.365 +
   5.366 +		obj->set_bone_matrix(assimp_matrix(aibone->mOffsetMatrix));
   5.367 +		mesh->add_bone(obj);
   5.368 +
   5.369 +		info_log("adding bone: %s\n", obj->get_name());
   5.370 +	}
   5.371 +
   5.372 +	return true;
   5.373 +}
   5.374 +
   5.375 +static Vector3 assimp_vector(const aiVector3D &v)
   5.376 +{
   5.377 +	return Vector3(v[0], v[1], v[2]);
   5.378 +}
   5.379 +
   5.380 +static Quaternion assimp_quat(const aiQuaternion &q)
   5.381 +{
   5.382 +	return Quaternion(q.w, Vector3(q.x, q.y, q.z));
   5.383 +}
   5.384 +
   5.385 +static Matrix4x4 assimp_matrix(const aiMatrix4x4 &aim)
   5.386 +{
   5.387 +	Matrix4x4 m;
   5.388 +	memcpy(m[0], &aim, 16 * sizeof(float));
   5.389 +	return m;
   5.390 +}
   5.391 +
   5.392 +/* convert an assimp keyframe time (ticks) into milliseconds */
   5.393 +static long assimp_time(const aiAnimation *anim, double aitime)
   5.394 +{
   5.395 +	double sec;
   5.396 +	if(anim->mTicksPerSecond < 1e-6) {
   5.397 +		// assume time is in frames?
   5.398 +		sec = aitime / 30.0;
   5.399 +	} else {
   5.400 +		sec = aitime / anim->mTicksPerSecond;
   5.401 +	}
   5.402 +	return (long)(sec * 1000.0);
   5.403 +}
   5.404 +
   5.405 +static void print_hierarchy(const aiNode *node)
   5.406 +{
   5.407 +	static int lvl;
   5.408 +	static int lvlopen[256];
   5.409 +
   5.410 +	for(int i=0; i<lvl; i++) {
   5.411 +		putchar(' ');
   5.412 +		if(lvlopen[i]) {
   5.413 +			putchar(i >= lvl - 1 ? '+' : '|');
   5.414 +		} else {
   5.415 +			putchar(i >= lvl - 1 ? '+' : ' ');
   5.416 +		}
   5.417 +	}
   5.418 +	info_log("- \"%s\"\n", node->mName.data);
   5.419 +
   5.420 +	lvlopen[lvl] = 1;
   5.421 +
   5.422 +	lvl++;
   5.423 +	for(unsigned int i=0; i<node->mNumChildren; i++) {
   5.424 +		if(i == node->mNumChildren - 1) {
   5.425 +			lvlopen[lvl - 1] = 0;
   5.426 +		}
   5.427 +		print_hierarchy(node->mChildren[i]);
   5.428 +	}
   5.429 +	lvl--;
   5.430 +}
   5.431 +
   5.432 +#else	// !defined USE_ASSIMP
   5.433 +
   5.434 +bool load_ass(Scene *scn, const char *fname)
   5.435 +{
   5.436 +	error_log("load_ass: assimp support not compiled in\n");
   5.437 +	return false;
   5.438 +}
   5.439 +
   5.440 +#endif
     6.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     6.2 +++ b/src/assload.h	Sun Aug 24 09:41:24 2014 +0300
     6.3 @@ -0,0 +1,9 @@
     6.4 +#ifndef ASSLOAD_H_
     6.5 +#define ASSLOAD_H_
     6.6 +
     6.7 +#include <vector>
     6.8 +#include "scene.h"
     6.9 +
    6.10 +bool load_ass(Scene *scn, const char *fname);
    6.11 +
    6.12 +#endif	// ASSLOAD_H_
     7.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     7.2 +++ b/src/dataset.h	Sun Aug 24 09:41:24 2014 +0300
     7.3 @@ -0,0 +1,44 @@
     7.4 +/** DataSet is a generic resource database with fast O(logn) lookups by name
     7.5 + * it can be used for texture managers, mesh managers, sound effect managers etc
     7.6 + *
     7.7 + * The constructor takes a load function and a destructor function to be called
     7.8 + * when a nonexistent resource is requested and needs to be loaded, and when
     7.9 + * the DataSet is destroyed. The destructor is optional and can be set to null
    7.10 + * if not needed.
    7.11 + *
    7.12 + * Requesting a resource works by simply calling get, example:
    7.13 + * ----------------------------------------------------------
    7.14 + * \code
    7.15 + * Texture *load_texture(const char *fname);
    7.16 + * void free_texture(Texture *tex);
    7.17 + *
    7.18 + * DataSet<Texture*> texman(load_texture, free_texture);
    7.19 + * Texture *foo = texman.get("foo.png");
    7.20 + * \endcode
    7.21 + */
    7.22 +#ifndef DATASET_H_
    7.23 +#define DATASET_H_
    7.24 +
    7.25 +#include <string>
    7.26 +#include <map>
    7.27 +
    7.28 +template <typename T>
    7.29 +class DataSet {
    7.30 +protected:
    7.31 +	mutable std::map<std::string, T> data;
    7.32 +
    7.33 +	T (*load)(const char*);
    7.34 +	void (*destroy)(T);
    7.35 +
    7.36 +public:
    7.37 +	DataSet(T (*load_func)(const char*), void (*destr_func)(T) = 0);
    7.38 +	~DataSet();
    7.39 +
    7.40 +	void clear();
    7.41 +
    7.42 +	T get(const char *name) const;
    7.43 +};
    7.44 +
    7.45 +#include "dataset.inl"
    7.46 +
    7.47 +#endif	// DATASET_H_
     8.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     8.2 +++ b/src/dataset.inl	Sun Aug 24 09:41:24 2014 +0300
     8.3 @@ -0,0 +1,55 @@
     8.4 +#include <stdio.h>
     8.5 +#include <string.h>
     8.6 +
     8.7 +template <typename T>
     8.8 +DataSet<T>::DataSet(T (*load_func)(const char*), void (*destr_func)(T))
     8.9 +{
    8.10 +	load = load_func;
    8.11 +	destroy = destr_func;
    8.12 +}
    8.13 +
    8.14 +template <typename T>
    8.15 +DataSet<T>::~DataSet()
    8.16 +{
    8.17 +	clear();
    8.18 +}
    8.19 +
    8.20 +template <typename T>
    8.21 +void DataSet<T>::clear()
    8.22 +{
    8.23 +	if(destroy) {
    8.24 +		typename std::map<std::string, T>::iterator it = data.begin();
    8.25 +		while(it != data.end()) {
    8.26 +			destroy(it++->second);
    8.27 +		}
    8.28 +	}
    8.29 +	data.clear();
    8.30 +}
    8.31 +
    8.32 +template <typename T>
    8.33 +T DataSet<T>::get(const char *name) const
    8.34 +{
    8.35 +	typename std::map<std::string, T>::const_iterator iter = data.find(name);
    8.36 +	if(iter != data.end()) {
    8.37 +		return iter->second;
    8.38 +	}
    8.39 +
    8.40 +	const char *fname, *slash;
    8.41 +	if((slash = strrchr(name, '/'))) {
    8.42 +		fname = slash + 1;
    8.43 +	} else {
    8.44 +		fname = name;
    8.45 +	}
    8.46 +
    8.47 +	std::string path = fname;
    8.48 +	if(path.empty()) {
    8.49 +		fprintf(stderr, "can't find data file: %s\n", name);
    8.50 +		return 0;
    8.51 +	}
    8.52 +
    8.53 +	T res = load(path.c_str());
    8.54 +	if(res) {
    8.55 +		data[name] = res;
    8.56 +	}
    8.57 +	return res;
    8.58 +}
     9.1 --- a/src/game.cc	Sat Aug 23 12:03:29 2014 +0300
     9.2 +++ b/src/game.cc	Sun Aug 24 09:41:24 2014 +0300
     9.3 @@ -1,4 +1,5 @@
     9.4  #include "game.h"
     9.5 +#include "gameopt.h"
     9.6  #include "opengl.h"
     9.7  #include "camera.h"
     9.8  #include "texture.h"
     9.9 @@ -18,11 +19,18 @@
    9.10  static FlyCamera cam;
    9.11  static Texture floor_tex;
    9.12  static bool keystate[256];
    9.13 +static float player_height = 1.68;
    9.14  
    9.15  bool game_init()
    9.16  {
    9.17 -	vr_init();
    9.18 -	//vr_use_module_named("null");
    9.19 +	if(opt.vr) {
    9.20 +		vr_init();
    9.21 +		if(opt.vr_module) {
    9.22 +			vr_use_module_named(opt.vr_module);
    9.23 +		}
    9.24 +
    9.25 +		player_height = vr_get_optf(VR_OPT_EYE_HEIGHT);
    9.26 +	}
    9.27  
    9.28  	glEnable(GL_DEPTH_TEST);
    9.29  	glEnable(GL_CULL_FACE);
    9.30 @@ -37,14 +45,16 @@
    9.31  		return false;
    9.32  	}
    9.33  
    9.34 -	cam.input_move(0, 0, 5);
    9.35 +	cam.input_move(0, player_height, 0);
    9.36  	return true;
    9.37  }
    9.38  
    9.39  void game_cleanup()
    9.40  {
    9.41  	floor_tex.destroy();
    9.42 -	vr_shutdown();
    9.43 +	if(opt.vr) {
    9.44 +		vr_shutdown();
    9.45 +	}
    9.46  
    9.47  	if(fbo) {
    9.48  		glDeleteFramebuffers(1, &fbo);
    9.49 @@ -88,24 +98,30 @@
    9.50  
    9.51  void game_render()
    9.52  {
    9.53 -	glBindFramebuffer(GL_FRAMEBUFFER, fbo);
    9.54 -	glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
    9.55 +	if(opt.vr) {
    9.56 +		glBindFramebuffer(GL_FRAMEBUFFER, fbo);
    9.57 +		glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
    9.58  
    9.59 -	glViewport(0, 0, rtwidth / 2.0, rtheight);
    9.60 -	vr_begin(VR_EYE_LEFT);
    9.61 -	game_render_eye(-1);
    9.62 -	vr_end();
    9.63 +		glViewport(0, 0, rtwidth / 2.0, rtheight);
    9.64 +		vr_begin(VR_EYE_LEFT);
    9.65 +		game_render_eye(-1);
    9.66 +		vr_end();
    9.67  
    9.68 -	glViewport(rtwidth / 2, 0, rtwidth / 2.0, rtheight);
    9.69 -	vr_begin(VR_EYE_RIGHT);
    9.70 -	game_render_eye(1);
    9.71 -	vr_end();
    9.72 +		glViewport(rtwidth / 2, 0, rtwidth / 2.0, rtheight);
    9.73 +		vr_begin(VR_EYE_RIGHT);
    9.74 +		game_render_eye(1);
    9.75 +		vr_end();
    9.76  
    9.77 -	glBindFramebuffer(GL_FRAMEBUFFER, 0);
    9.78 -	glViewport(0, 0, fb_width, fb_height);
    9.79 +		glBindFramebuffer(GL_FRAMEBUFFER, 0);
    9.80 +		glViewport(0, 0, fb_width, fb_height);
    9.81  
    9.82 -	vr_output_texture(rtarg->get_texture_id(), 0, 0, (float)rtwidth / (float)rtarg->get_width(),
    9.83 -		(float)rtheight / (float)rtarg->get_height());
    9.84 +		vr_output_texture(rtarg->get_texture_id(), 0, 0, (float)rtwidth / (float)rtarg->get_width(),
    9.85 +			(float)rtheight / (float)rtarg->get_height());
    9.86 +	} else {
    9.87 +		glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
    9.88 +
    9.89 +		game_render_eye(0);
    9.90 +	}
    9.91  
    9.92  	vr_swap_buffers();
    9.93  }
    9.94 @@ -218,14 +234,11 @@
    9.95  
    9.96  static void draw_scene()
    9.97  {
    9.98 -	glMatrixMode(GL_MODELVIEW);
    9.99 -	glTranslatef(0, -1.5, 0);
   9.100 -
   9.101  	float lpos[] = {-20, 30, 10, 1};
   9.102  	glLightfv(GL_LIGHT0, GL_POSITION, lpos);
   9.103  
   9.104  	glEnable(GL_TEXTURE_2D);
   9.105 -	floor_tex.bind();
   9.106 +	bind_texture(&floor_tex);
   9.107  
   9.108  	glMatrixMode(GL_TEXTURE);
   9.109  	glScalef(8, 8, 8);
   9.110 @@ -251,19 +264,19 @@
   9.111  		glNormal3f(0, 1, 1);
   9.112  		glVertex3f(-1, 0, 1);
   9.113  		glVertex3f(1, 0, 1);
   9.114 -		glVertex3f(0, 1.2, 0);
   9.115 +		glVertex3f(0, 1.75, 0);
   9.116  		glNormal3f(1, 1, 0);
   9.117  		glVertex3f(1, 0, 1);
   9.118  		glVertex3f(1, 0, -1);
   9.119 -		glVertex3f(0, 1.2, 0);
   9.120 +		glVertex3f(0, 1.75, 0);
   9.121  		glNormal3f(0, 1, -1);
   9.122  		glVertex3f(1, 0, -1);
   9.123  		glVertex3f(-1, 0, -1);
   9.124 -		glVertex3f(0, 1.2, 0);
   9.125 +		glVertex3f(0, 1.75, 0);
   9.126  		glNormal3f(-1, 1, 0);
   9.127  		glVertex3f(-1, 0, -1);
   9.128  		glVertex3f(-1, 0, 1);
   9.129 -		glVertex3f(0, 1.2, 0);
   9.130 +		glVertex3f(0, 1.75, 0);
   9.131  		glEnd();
   9.132  
   9.133  		glPopMatrix();
    10.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    10.2 +++ b/src/gameopt.cc	Sun Aug 24 09:41:24 2014 +0300
    10.3 @@ -0,0 +1,39 @@
    10.4 +#include <stdio.h>
    10.5 +#include <string.h>
    10.6 +#include "gameopt.h"
    10.7 +
    10.8 +GameOption opt;
    10.9 +
   10.10 +bool parse_args(int argc, char **argv)
   10.11 +{
   10.12 +	int i;
   10.13 +
   10.14 +	for(i=1; i<argc; i++) {
   10.15 +		if(argv[i][0] == '-') {
   10.16 +			if(strcmp(argv[i], "-vr") == 0) {
   10.17 +				opt.vr = true;
   10.18 +
   10.19 +			} else if(strcmp(argv[i], "-vrmodule") == 0) {
   10.20 +				opt.vr_module = argv[++i];
   10.21 +				if(strcmp(opt.vr_module, "stereo") == 0) {
   10.22 +					opt.stereo = true;
   10.23 +				}
   10.24 +
   10.25 +			} else {
   10.26 +				fprintf(stderr, "invalid option: %s\n", argv[i]);
   10.27 +				return false;
   10.28 +			}
   10.29 +		} else {
   10.30 +			fprintf(stderr, "unexpected argument: %s\n", argv[i]);
   10.31 +			return false;
   10.32 +		}
   10.33 +	}
   10.34 +	return true;
   10.35 +}
   10.36 +
   10.37 +GameOption::GameOption()
   10.38 +{
   10.39 +	vr = false;
   10.40 +	vr_module = 0;
   10.41 +	stereo = false;
   10.42 +}
    11.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    11.2 +++ b/src/gameopt.h	Sun Aug 24 09:41:24 2014 +0300
    11.3 @@ -0,0 +1,16 @@
    11.4 +#ifndef GAMEOPT_H_
    11.5 +#define GAMEOPT_H_
    11.6 +
    11.7 +struct GameOption {
    11.8 +	bool vr;
    11.9 +	char *vr_module;
   11.10 +	bool stereo;
   11.11 +
   11.12 +	GameOption();
   11.13 +};
   11.14 +
   11.15 +extern GameOption opt;
   11.16 +
   11.17 +bool parse_args(int argc, char **argv);
   11.18 +
   11.19 +#endif	// GAMEOPT_H_
    12.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    12.2 +++ b/src/geom.cc	Sun Aug 24 09:41:24 2014 +0300
    12.3 @@ -0,0 +1,270 @@
    12.4 +#include <algorithm>
    12.5 +#include <float.h>
    12.6 +#include "geom.h"
    12.7 +#include "logger.h"
    12.8 +
    12.9 +GeomObject::~GeomObject()
   12.10 +{
   12.11 +}
   12.12 +
   12.13 +
   12.14 +Sphere::Sphere()
   12.15 +{
   12.16 +	radius = 1.0;
   12.17 +}
   12.18 +
   12.19 +Sphere::Sphere(const Vector3 &cent, float radius)
   12.20 +	: center(cent)
   12.21 +{
   12.22 +	this->radius = radius;
   12.23 +}
   12.24 +
   12.25 +void Sphere::set_union(const GeomObject *obj1, const GeomObject *obj2)
   12.26 +{
   12.27 +	const Sphere *sph1 = dynamic_cast<const Sphere*>(obj1);
   12.28 +	const Sphere *sph2 = dynamic_cast<const Sphere*>(obj2);
   12.29 +
   12.30 +	if(!sph1 || !sph2) {
   12.31 +		error_log("Sphere::set_union: arguments must be spheres");
   12.32 +		return;
   12.33 +	}
   12.34 +
   12.35 +	float dist = (sph1->center - sph2->center).length();
   12.36 +	float surf_dist = dist - (sph1->radius + sph2->radius);
   12.37 +	float d1 = sph1->radius + surf_dist / 2.0;
   12.38 +	float d2 = sph2->radius + surf_dist / 2.0;
   12.39 +	float t = d1 / (d1 + d2);
   12.40 +
   12.41 +	if(t < 0.0) t = 0.0;
   12.42 +	if(t > 1.0) t = 1.0;
   12.43 +
   12.44 +	center = sph1->center * t + sph2->center * (1.0 - t);
   12.45 +	radius = std::max(dist * t + sph2->radius, dist * (1.0f - t) + sph1->radius);
   12.46 +}
   12.47 +
   12.48 +void Sphere::set_intersection(const GeomObject *obj1, const GeomObject *obj2)
   12.49 +{
   12.50 +	error_log("Sphere::intersection undefined\n");
   12.51 +}
   12.52 +
   12.53 +bool Sphere::contains(const Vector3 &pt) const
   12.54 +{
   12.55 +	float dist_sq = (pt - center).length_sq();
   12.56 +	return dist_sq <= radius * radius;
   12.57 +}
   12.58 +
   12.59 +bool Sphere::intersect(const Ray &ray, HitPoint *hit) const
   12.60 +{
   12.61 +	float a = dot_product(ray.dir, ray.dir);
   12.62 +	float b = 2.0 * ray.dir.x * (ray.origin.x - center.x) +
   12.63 +		2.0 * ray.dir.y * (ray.origin.y - center.y) +
   12.64 +		2.0 * ray.dir.z * (ray.origin.z - center.z);
   12.65 +	float c = dot_product(ray.origin, ray.origin) + dot_product(center, center) -
   12.66 +		2.0 * dot_product(ray.origin, center) - radius * radius;
   12.67 +
   12.68 +	float discr = b * b - 4.0 * a * c;
   12.69 +	if(discr < 1e-4) {
   12.70 +		return false;
   12.71 +	}
   12.72 +
   12.73 +	float sqrt_discr = sqrt(discr);
   12.74 +	float t0 = (-b + sqrt_discr) / (2.0 * a);
   12.75 +	float t1 = (-b - sqrt_discr) / (2.0 * a);
   12.76 +
   12.77 +	if(t0 < 1e-4)
   12.78 +		t0 = t1;
   12.79 +	if(t1 < 1e-4)
   12.80 +		t1 = t0;
   12.81 +
   12.82 +	float t = t0 < t1 ? t0 : t1;
   12.83 +	if(t < 1e-4) {
   12.84 +		return false;
   12.85 +	}
   12.86 +
   12.87 +	// fill the HitPoint structure
   12.88 +	if(hit) {
   12.89 +		hit->obj = this;
   12.90 +		hit->dist = t;
   12.91 +		hit->pos = ray.origin + ray.dir * t;
   12.92 +		hit->normal = (hit->pos - center) / radius;
   12.93 +	}
   12.94 +	return true;
   12.95 +}
   12.96 +
   12.97 +
   12.98 +AABox::AABox()
   12.99 +{
  12.100 +}
  12.101 +
  12.102 +AABox::AABox(const Vector3 &vmin, const Vector3 &vmax)
  12.103 +	: min(vmin), max(vmax)
  12.104 +{
  12.105 +}
  12.106 +
  12.107 +void AABox::set_union(const GeomObject *obj1, const GeomObject *obj2)
  12.108 +{
  12.109 +	const AABox *box1 = dynamic_cast<const AABox*>(obj1);
  12.110 +	const AABox *box2 = dynamic_cast<const AABox*>(obj2);
  12.111 +
  12.112 +	if(!box1 || !box2) {
  12.113 +		error_log("AABox::set_union: arguments must be AABoxes too\n");
  12.114 +		return;
  12.115 +	}
  12.116 +
  12.117 +	min.x = std::min(box1->min.x, box2->min.x);
  12.118 +	min.y = std::min(box1->min.y, box2->min.y);
  12.119 +	min.z = std::min(box1->min.z, box2->min.z);
  12.120 +
  12.121 +	max.x = std::max(box1->max.x, box2->max.x);
  12.122 +	max.y = std::max(box1->max.y, box2->max.y);
  12.123 +	max.z = std::max(box1->max.z, box2->max.z);
  12.124 +}
  12.125 +
  12.126 +void AABox::set_intersection(const GeomObject *obj1, const GeomObject *obj2)
  12.127 +{
  12.128 +	const AABox *box1 = dynamic_cast<const AABox*>(obj1);
  12.129 +	const AABox *box2 = dynamic_cast<const AABox*>(obj2);
  12.130 +
  12.131 +	if(!box1 || !box2) {
  12.132 +		error_log("AABox::set_intersection: arguments must be AABoxes too\n");
  12.133 +		return;
  12.134 +	}
  12.135 +
  12.136 +	for(int i=0; i<3; i++) {
  12.137 +		min[i] = std::max(box1->min[i], box2->min[i]);
  12.138 +		max[i] = std::min(box1->max[i], box2->max[i]);
  12.139 +
  12.140 +		if(max[i] < min[i]) {
  12.141 +			max[i] = min[i];
  12.142 +		}
  12.143 +	}
  12.144 +}
  12.145 +
  12.146 +bool AABox::contains(const Vector3 &pt) const
  12.147 +{
  12.148 +	return pt.x >= min.x && pt.x <= max.x &&
  12.149 +		pt.y >= min.y && pt.y <= max.y &&
  12.150 +		pt.z >= min.z && pt.z <= max.z;
  12.151 +}
  12.152 +
  12.153 +bool AABox::intersect(const Ray &ray, HitPoint *hit) const
  12.154 +{
  12.155 +	Vector3 param[2] = {min, max};
  12.156 +	Vector3 inv_dir(1.0 / ray.dir.x, 1.0 / ray.dir.y, 1.0 / ray.dir.z);
  12.157 +	int sign[3] = {inv_dir.x < 0, inv_dir.y < 0, inv_dir.z < 0};
  12.158 +
  12.159 +	float tmin = (param[sign[0]].x - ray.origin.x) * inv_dir.x;
  12.160 +	float tmax = (param[1 - sign[0]].x - ray.origin.x) * inv_dir.x;
  12.161 +	float tymin = (param[sign[1]].y - ray.origin.y) * inv_dir.y;
  12.162 +	float tymax = (param[1 - sign[1]].y - ray.origin.y) * inv_dir.y;
  12.163 +
  12.164 +	if(tmin > tymax || tymin > tmax) {
  12.165 +		return false;
  12.166 +	}
  12.167 +	if(tymin > tmin) {
  12.168 +		tmin = tymin;
  12.169 +	}
  12.170 +	if(tymax < tmax) {
  12.171 +		tmax = tymax;
  12.172 +	}
  12.173 +
  12.174 +	float tzmin = (param[sign[2]].z - ray.origin.z) * inv_dir.z;
  12.175 +	float tzmax = (param[1 - sign[2]].z - ray.origin.z) * inv_dir.z;
  12.176 +
  12.177 +	if(tmin > tzmax || tzmin > tmax) {
  12.178 +		return false;
  12.179 +	}
  12.180 +	if(tzmin > tmin) {
  12.181 +		tmin = tzmin;
  12.182 +	}
  12.183 +	if(tzmax < tmax) {
  12.184 +		tmax = tzmax;
  12.185 +	}
  12.186 +
  12.187 +	float t = tmin < 1e-4 ? tmax : tmin;
  12.188 +	if(t >= 1e-4) {
  12.189 +
  12.190 +		if(hit) {
  12.191 +			hit->obj = this;
  12.192 +			hit->dist = t;
  12.193 +			hit->pos = ray.origin + ray.dir * t;
  12.194 +
  12.195 +			float min_dist = FLT_MAX;
  12.196 +			Vector3 offs = min + (max - min) / 2.0;
  12.197 +			Vector3 local_hit = hit->pos - offs;
  12.198 +
  12.199 +			static const Vector3 axis[] = {
  12.200 +				Vector3(1, 0, 0), Vector3(0, 1, 0), Vector3(0, 0, 1)
  12.201 +			};
  12.202 +			//int tcidx[][2] = {{2, 1}, {0, 2}, {0, 1}};
  12.203 +
  12.204 +			for(int i=0; i<3; i++) {
  12.205 +				float dist = fabs((max[i] - offs[i]) - fabs(local_hit[i]));
  12.206 +				if(dist < min_dist) {
  12.207 +					min_dist = dist;
  12.208 +					hit->normal = axis[i] * (local_hit[i] < 0.0 ? 1.0 : -1.0);
  12.209 +					//hit->texcoord = Vector2(hit->pos[tcidx[i][0]], hit->pos[tcidx[i][1]]);
  12.210 +				}
  12.211 +			}
  12.212 +		}
  12.213 +		return true;
  12.214 +	}
  12.215 +	return false;
  12.216 +
  12.217 +}
  12.218 +
  12.219 +Plane::Plane()
  12.220 +	: normal(0.0, 1.0, 0.0)
  12.221 +{
  12.222 +}
  12.223 +
  12.224 +Plane::Plane(const Vector3 &p, const Vector3 &norm)
  12.225 +	: pt(p)
  12.226 +{
  12.227 +	normal = norm.normalized();
  12.228 +}
  12.229 +
  12.230 +Plane::Plane(const Vector3 &p1, const Vector3 &p2, const Vector3 &p3)
  12.231 +	: pt(p1)
  12.232 +{
  12.233 +	normal = cross_product(p2 - p1, p3 - p1).normalized();
  12.234 +}
  12.235 +
  12.236 +Plane::Plane(const Vector3 &normal, float dist)
  12.237 +{
  12.238 +	this->normal = normal.normalized();
  12.239 +	pt = this->normal * dist;
  12.240 +}
  12.241 +
  12.242 +void Plane::set_union(const GeomObject *obj1, const GeomObject *obj2)
  12.243 +{
  12.244 +	error_log("Plane::set_union undefined\n");
  12.245 +}
  12.246 +
  12.247 +void Plane::set_intersection(const GeomObject *obj1, const GeomObject *obj2)
  12.248 +{
  12.249 +	error_log("Plane::set_intersection undefined\n");
  12.250 +}
  12.251 +
  12.252 +bool Plane::contains(const Vector3 &pt) const
  12.253 +{
  12.254 +	return false;	// TODO: maybe define containment as half-space containment?
  12.255 +}
  12.256 +
  12.257 +bool Plane::intersect(const Ray &ray, HitPoint *hit) const
  12.258 +{
  12.259 +	float ndotdir = dot_product(normal, ray.dir);
  12.260 +	if(fabs(ndotdir) < 1e-4) {
  12.261 +		return false;
  12.262 +	}
  12.263 +
  12.264 +	if(hit) {
  12.265 +		Vector3 ptdir = pt - ray.origin;
  12.266 +		float t = dot_product(normal, ptdir) / ndotdir;
  12.267 +
  12.268 +		hit->pos = ray.origin + ray.dir * t;
  12.269 +		hit->normal = normal;
  12.270 +		hit->obj = this;
  12.271 +	}
  12.272 +	return true;
  12.273 +}
    13.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    13.2 +++ b/src/geom.h	Sun Aug 24 09:41:24 2014 +0300
    13.3 @@ -0,0 +1,71 @@
    13.4 +#ifndef GEOMOBJ_H_
    13.5 +#define GEOMOBJ_H_
    13.6 +
    13.7 +#include "vmath/vmath.h"
    13.8 +
    13.9 +class GeomObject;
   13.10 +
   13.11 +struct HitPoint {
   13.12 +	float dist;				//< parametric distance along the ray
   13.13 +	Vector3 pos;			//< position of intersection (orig + dir * dist)
   13.14 +	Vector3 normal;			//< normal at the point of intersection
   13.15 +	const void *obj;		//< pointer to the intersected object
   13.16 +};
   13.17 +
   13.18 +class GeomObject {
   13.19 +public:
   13.20 +	virtual ~GeomObject();
   13.21 +
   13.22 +	virtual void set_union(const GeomObject *obj1, const GeomObject *obj2) = 0;
   13.23 +	virtual void set_intersection(const GeomObject *obj1, const GeomObject *obj2) = 0;
   13.24 +
   13.25 +	virtual bool contains(const Vector3 &pt) const = 0;
   13.26 +	virtual bool intersect(const Ray &ray, HitPoint *hit = 0) const = 0;
   13.27 +};
   13.28 +
   13.29 +class Sphere : public GeomObject {
   13.30 +public:
   13.31 +	Vector3 center;
   13.32 +	float radius;
   13.33 +
   13.34 +	Sphere();
   13.35 +	Sphere(const Vector3 &center, float radius);
   13.36 +
   13.37 +	void set_union(const GeomObject *obj1, const GeomObject *obj2);
   13.38 +	void set_intersection(const GeomObject *obj1, const GeomObject *obj2);
   13.39 +
   13.40 +	bool contains(const Vector3 &pt) const;
   13.41 +	bool intersect(const Ray &ray, HitPoint *hit = 0) const;
   13.42 +};
   13.43 +
   13.44 +class AABox : public GeomObject {
   13.45 +public:
   13.46 +	Vector3 min, max;
   13.47 +
   13.48 +	AABox();
   13.49 +	AABox(const Vector3 &min, const Vector3 &max);
   13.50 +
   13.51 +	void set_union(const GeomObject *obj1, const GeomObject *obj2);
   13.52 +	void set_intersection(const GeomObject *obj1, const GeomObject *obj2);
   13.53 +
   13.54 +	bool contains(const Vector3 &pt) const;
   13.55 +	bool intersect(const Ray &ray, HitPoint *hit = 0) const;
   13.56 +};
   13.57 +
   13.58 +class Plane : public GeomObject {
   13.59 +public:
   13.60 +	Vector3 pt, normal;
   13.61 +
   13.62 +	Plane();
   13.63 +	Plane(const Vector3 &pt, const Vector3 &normal);
   13.64 +	Plane(const Vector3 &p1, const Vector3 &p2, const Vector3 &p3);
   13.65 +	Plane(const Vector3 &normal, float dist);
   13.66 +
   13.67 +	void set_union(const GeomObject *obj1, const GeomObject *obj2);
   13.68 +	void set_intersection(const GeomObject *obj1, const GeomObject *obj2);
   13.69 +
   13.70 +	bool contains(const Vector3 &pt) const;
   13.71 +	bool intersect(const Ray &ray, HitPoint *hit = 0) const;
   13.72 +};
   13.73 +
   13.74 +#endif	// GEOMOBJ_H_
    14.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    14.2 +++ b/src/light.cc	Sun Aug 24 09:41:24 2014 +0300
    14.3 @@ -0,0 +1,36 @@
    14.4 +#include "light.h"
    14.5 +
    14.6 +Light::Light()
    14.7 +{
    14.8 +	cast_shadows = false;
    14.9 +}
   14.10 +
   14.11 +void Light::set_color(const Vector3 &color)
   14.12 +{
   14.13 +	this->color = color;
   14.14 +}
   14.15 +
   14.16 +const Vector3 &Light::get_color() const
   14.17 +{
   14.18 +	return color;
   14.19 +}
   14.20 +
   14.21 +void Light::set_attenuation(const Vector3 &att)
   14.22 +{
   14.23 +	attenuation = att;
   14.24 +}
   14.25 +
   14.26 +const Vector3 &Light::get_attenuation() const
   14.27 +{
   14.28 +	return attenuation;
   14.29 +}
   14.30 +
   14.31 +void Light::set_shadow_caster(bool s)
   14.32 +{
   14.33 +	cast_shadows = s;
   14.34 +}
   14.35 +
   14.36 +bool Light::is_shadow_caster() const
   14.37 +{
   14.38 +	return cast_shadows;
   14.39 +}
    15.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    15.2 +++ b/src/light.h	Sun Aug 24 09:41:24 2014 +0300
    15.3 @@ -0,0 +1,27 @@
    15.4 +#ifndef GOATGFX_LIGHT_H_
    15.5 +#define GOATGFX_LIGHT_H_
    15.6 +
    15.7 +#include "xform_node.h"
    15.8 +
    15.9 +
   15.10 +class Light : public XFormNode {
   15.11 +private:
   15.12 +	Vector3 color;
   15.13 +	Vector3 attenuation;
   15.14 +
   15.15 +	bool cast_shadows;
   15.16 +
   15.17 +public:
   15.18 +	Light();
   15.19 +
   15.20 +	void set_color(const Vector3 &color);
   15.21 +	const Vector3 &get_color() const;
   15.22 +
   15.23 +	void set_attenuation(const Vector3 &att);
   15.24 +	const Vector3 &get_attenuation() const;
   15.25 +
   15.26 +	void set_shadow_caster(bool s);
   15.27 +	bool is_shadow_caster() const;
   15.28 +};
   15.29 +
   15.30 +#endif	// GOATGFX_LIGHT_H_
    16.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    16.2 +++ b/src/logger.cc	Sun Aug 24 09:41:24 2014 +0300
    16.3 @@ -0,0 +1,118 @@
    16.4 +#include <stdio.h>
    16.5 +#include <stdarg.h>
    16.6 +#include "logger.h"
    16.7 +
    16.8 +#if defined(unix) || defined(__unix__) || defined(__APPLE__)
    16.9 +#include <unistd.h>
   16.10 +#elif defined(WIN32)
   16.11 +#include <windows.h>
   16.12 +#endif
   16.13 +
   16.14 +enum { LOG_INFO, LOG_WARNING, LOG_ERROR, LOG_FATAL, LOG_DEBUG };
   16.15 +
   16.16 +static int typecolor(int type);
   16.17 +
   16.18 +static FILE *fp = stdout;
   16.19 +
   16.20 +static void logmsg(int type, const char *fmt, va_list ap)
   16.21 +{
   16.22 +#if defined(unix) || defined(__unix__) || (defined(__APPLE__) && !defined(TARGET_IPHONE))
   16.23 +	if(isatty(fileno(fp)) && type != LOG_INFO) {
   16.24 +		int c = typecolor(type);
   16.25 +		fprintf(fp, "\033[%dm", c);
   16.26 +		vfprintf(fp, fmt, ap);
   16.27 +		fprintf(fp, "\033[0m");
   16.28 +	} else
   16.29 +#endif
   16.30 +	{
   16.31 +		vfprintf(fp, fmt, ap);
   16.32 +	}
   16.33 +	if(type == LOG_ERROR || type == LOG_FATAL || type == LOG_DEBUG) {
   16.34 +		fflush(fp);
   16.35 +	}
   16.36 +
   16.37 +#ifdef WIN32
   16.38 +	if(type == LOG_FATAL) {
   16.39 +		static char msgbuf[1024];
   16.40 +		vsnprintf(msgbuf, sizeof msgbuf - 1, fmt, ap);
   16.41 +		msgbuf[sizeof msgbuf - 1] = 0;
   16.42 +		MessageBox(0, msgbuf, "Fatal error", MB_OK | MB_ICONSTOP);
   16.43 +	}
   16.44 +#endif
   16.45 +}
   16.46 +
   16.47 +void info_log(const char *fmt, ...)
   16.48 +{
   16.49 +	va_list ap;
   16.50 +
   16.51 +	va_start(ap, fmt);
   16.52 +	logmsg(LOG_INFO, fmt, ap);
   16.53 +	va_end(ap);
   16.54 +}
   16.55 +
   16.56 +void warning_log(const char *fmt, ...)
   16.57 +{
   16.58 +	va_list ap;
   16.59 +
   16.60 +	va_start(ap, fmt);
   16.61 +	logmsg(LOG_WARNING, fmt, ap);
   16.62 +	va_end(ap);
   16.63 +}
   16.64 +
   16.65 +void error_log(const char *fmt, ...)
   16.66 +{
   16.67 +	va_list ap;
   16.68 +
   16.69 +	va_start(ap, fmt);
   16.70 +	logmsg(LOG_ERROR, fmt, ap);
   16.71 +	va_end(ap);
   16.72 +}
   16.73 +
   16.74 +void fatal_log(const char *fmt, ...)
   16.75 +{
   16.76 +	va_list ap;
   16.77 +
   16.78 +	va_start(ap, fmt);
   16.79 +	logmsg(LOG_FATAL, fmt, ap);
   16.80 +	va_end(ap);
   16.81 +}
   16.82 +
   16.83 +void debug_log(const char *fmt, ...)
   16.84 +{
   16.85 +	va_list ap;
   16.86 +
   16.87 +	va_start(ap, fmt);
   16.88 +	logmsg(LOG_DEBUG, fmt, ap);
   16.89 +	va_end(ap);
   16.90 +}
   16.91 +
   16.92 +enum {
   16.93 +	BLACK = 0,
   16.94 +	RED,
   16.95 +	GREEN,
   16.96 +	YELLOW,
   16.97 +	BLUE,
   16.98 +	MAGENTA,
   16.99 +	CYAN,
  16.100 +	WHITE
  16.101 +};
  16.102 +
  16.103 +#define ANSI_FGCOLOR(x)	(30 + (x))
  16.104 +#define ANSI_BGCOLOR(x)	(40 + (x))
  16.105 +
  16.106 +static int typecolor(int type)
  16.107 +{
  16.108 +	switch(type) {
  16.109 +	case LOG_ERROR:
  16.110 +		return ANSI_FGCOLOR(RED);
  16.111 +	case LOG_FATAL:
  16.112 +		return ANSI_FGCOLOR(RED);	// TODO differentiate from LOG_ERROR
  16.113 +	case LOG_WARNING:
  16.114 +		return ANSI_FGCOLOR(YELLOW);
  16.115 +	case LOG_DEBUG:
  16.116 +		return ANSI_FGCOLOR(MAGENTA);
  16.117 +	default:
  16.118 +		break;
  16.119 +	}
  16.120 +	return 37;
  16.121 +}
    17.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    17.2 +++ b/src/logger.h	Sun Aug 24 09:41:24 2014 +0300
    17.3 @@ -0,0 +1,10 @@
    17.4 +#ifndef LOGGER_H_
    17.5 +#define LOGGER_H_
    17.6 +
    17.7 +void info_log(const char *fmt, ...);
    17.8 +void warning_log(const char *fmt, ...);
    17.9 +void error_log(const char *fmt, ...);
   17.10 +void fatal_log(const char *fmt, ...);
   17.11 +void debug_log(const char *fmt, ...);
   17.12 +
   17.13 +#endif	// LOGGER_H_
    18.1 --- a/src/main.cc	Sat Aug 23 12:03:29 2014 +0300
    18.2 +++ b/src/main.cc	Sun Aug 24 09:41:24 2014 +0300
    18.3 @@ -2,6 +2,7 @@
    18.4  #include <stdlib.h>
    18.5  #include "opengl.h"
    18.6  #include "game.h"
    18.7 +#include "gameopt.h"
    18.8  #include "vr/vr.h"
    18.9  
   18.10  static bool init();
   18.11 @@ -23,8 +24,13 @@
   18.12  {
   18.13  	glutInitWindowSize(1024, 600);
   18.14  	glutInit(&argc, argv);
   18.15 -	glutInitDisplayMode(GLUT_RGB | GLUT_DEPTH | GLUT_DOUBLE);
   18.16 -	glutCreateWindow("VR Chess");
   18.17 +
   18.18 +	if(!parse_args(argc, argv)) {
   18.19 +		return 1;
   18.20 +	}
   18.21 +
   18.22 +	glutInitDisplayMode(GLUT_RGB | GLUT_DEPTH | GLUT_DOUBLE | (opt.stereo ? GLUT_STEREO : 0));
   18.23 +	glutCreateWindow("LD48 #30 - connected worlds");
   18.24  
   18.25  	glutDisplayFunc(display);
   18.26  	glutIdleFunc(idle);
   18.27 @@ -53,10 +59,12 @@
   18.28  		return false;
   18.29  	}
   18.30  
   18.31 -	int win_xsz = vr_get_opti(VR_OPT_DISPLAY_WIDTH);
   18.32 -	int win_ysz = vr_get_opti(VR_OPT_DISPLAY_HEIGHT);
   18.33 -	if(win_xsz && win_ysz) {
   18.34 -		glutReshapeWindow(win_xsz, win_ysz);
   18.35 +	if(opt.vr) {
   18.36 +		int win_xsz = vr_get_opti(VR_OPT_DISPLAY_WIDTH);
   18.37 +		int win_ysz = vr_get_opti(VR_OPT_DISPLAY_HEIGHT);
   18.38 +		if(win_xsz && win_ysz) {
   18.39 +			glutReshapeWindow(win_xsz, win_ysz);
   18.40 +		}
   18.41  	}
   18.42  	return true;
   18.43  }
   18.44 @@ -110,8 +118,8 @@
   18.45  			if(xoffs || yoffs) {
   18.46  				printf("repositioning: %d,%d\n", xoffs, yoffs);
   18.47  				glutPositionWindow(xoffs, yoffs);
   18.48 -				fullscreen_pending = true;
   18.49  			}
   18.50 +			fullscreen_pending = true;
   18.51  		} else {
   18.52  			fullscreen_pending = false;
   18.53  			glutPositionWindow(prev_xpos, prev_ypos);
    19.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    19.2 +++ b/src/material.cc	Sun Aug 24 09:41:24 2014 +0300
    19.3 @@ -0,0 +1,75 @@
    19.4 +#include "opengl.h"
    19.5 +#include "material.h"
    19.6 +#include "unistate.h"
    19.7 +
    19.8 +Material::Material()
    19.9 +	: diffuse(1, 1, 1), specular(0, 0, 0)
   19.10 +{
   19.11 +	alpha = 1.0;
   19.12 +	shininess = 1.0;
   19.13 +
   19.14 +	for(int i=0; i<MAX_MTL_TEXTURES; i++) {
   19.15 +		tex[i] = 0;
   19.16 +	}
   19.17 +}
   19.18 +
   19.19 +
   19.20 +void Material::setup(bool use_textures) const
   19.21 +{
   19.22 +	static bool done_init;
   19.23 +	static int st_diffuse_idx, st_specular_idx, st_shininess_idx, st_alpha_idx;
   19.24 +	static int st_tex_diffuse_idx, st_tex_specular_idx, st_tex_normal_idx;
   19.25 +	static int st_tex_idx[MAX_MTL_TEXTURES];
   19.26 +
   19.27 +	if(!done_init) {
   19.28 +		st_diffuse_idx = add_unistate("st_mtl_diffuse", ST_FLOAT3);
   19.29 +		st_specular_idx = add_unistate("st_mtl_specular", ST_FLOAT3);
   19.30 +		st_shininess_idx = add_unistate("st_mtl_shininess", ST_FLOAT);
   19.31 +		st_alpha_idx = add_unistate("st_mtl_alpha", ST_FLOAT);
   19.32 +		st_tex_diffuse_idx = add_unistate("st_tex_diffuse", ST_INT);
   19.33 +		st_tex_specular_idx = add_unistate("st_tex_specular", ST_INT);
   19.34 +		st_tex_normal_idx = add_unistate("st_tex_normal", ST_INT);
   19.35 +
   19.36 +		for(int i=0; i<MAX_MTL_TEXTURES; i++) {
   19.37 +			char name[32];
   19.38 +			sprintf(name, "st_tex%d", i);
   19.39 +			st_tex_idx[i] = add_unistate(name, ST_INT);
   19.40 +		}
   19.41 +		done_init = true;
   19.42 +	}
   19.43 +
   19.44 +	set_unistate(st_diffuse_idx, diffuse);
   19.45 +	set_unistate(st_specular_idx, specular);
   19.46 +	set_unistate(st_shininess_idx, shininess);
   19.47 +	set_unistate(st_alpha_idx, alpha);
   19.48 +
   19.49 +	if(!use_textures) {
   19.50 +		return;	// we're done
   19.51 +	}
   19.52 +
   19.53 +	const int tex_named_idx[] = {
   19.54 +		st_tex_diffuse_idx, st_tex_specular_idx, st_tex_normal_idx, -1
   19.55 +	};
   19.56 +
   19.57 +	int tex_unit = 0;
   19.58 +	for(int i=0; i<MAX_MTL_TEXTURES; i++) {
   19.59 +		if(tex[i]) {
   19.60 +			bind_texture(tex[i], tex_unit);
   19.61 +
   19.62 +			if(tex_named_idx[i] != -1) {
   19.63 +				set_unistate(tex_named_idx[i], tex_unit);
   19.64 +			}
   19.65 +			set_unistate(st_tex_idx[i], tex_unit);
   19.66 +			tex_unit++;
   19.67 +		}
   19.68 +	}
   19.69 +
   19.70 +	// also do fixed-function setup
   19.71 +	float dcol[] = {diffuse.x, diffuse.y, diffuse.z, alpha};
   19.72 +	glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, dcol);
   19.73 +
   19.74 +	float scol[] = {specular.x, specular.y, specular.z, 1.0f};
   19.75 +	glMaterialfv(GL_FRONT_AND_BACK, GL_SPECULAR, scol);
   19.76 +
   19.77 +	glMaterialf(GL_FRONT_AND_BACK, GL_SHININESS, shininess);
   19.78 +}
    20.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    20.2 +++ b/src/material.h	Sun Aug 24 09:41:24 2014 +0300
    20.3 @@ -0,0 +1,29 @@
    20.4 +#ifndef MATERIAL_H_
    20.5 +#define MATERIAL_H_
    20.6 +
    20.7 +#include "vmath/vmath.h"
    20.8 +#include "texture.h"
    20.9 +
   20.10 +enum {
   20.11 +	TEX_DIFFUSE,
   20.12 +	TEX_SPECULAR,
   20.13 +	TEX_NORMAL,
   20.14 +	TEX_MISC,
   20.15 +
   20.16 +	MAX_MTL_TEXTURES
   20.17 +};
   20.18 +
   20.19 +class Material {
   20.20 +public:
   20.21 +	Vector3 diffuse, specular;
   20.22 +	float alpha;
   20.23 +	float shininess;
   20.24 +
   20.25 +	Texture *tex[MAX_MTL_TEXTURES];
   20.26 +
   20.27 +	Material();
   20.28 +
   20.29 +	void setup(bool use_textures = true) const;
   20.30 +};
   20.31 +
   20.32 +#endif	// MATERIAL_H_
    21.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    21.2 +++ b/src/mesh.cc	Sun Aug 24 09:41:24 2014 +0300
    21.3 @@ -0,0 +1,1024 @@
    21.4 +#include <stdio.h>
    21.5 +#include <stdlib.h>
    21.6 +#include <float.h>
    21.7 +#include <assert.h>
    21.8 +#include "opengl.h"
    21.9 +#include "mesh.h"
   21.10 +#include "xform_node.h"
   21.11 +#include "shader.h"
   21.12 +#include "logger.h"
   21.13 +
   21.14 +int Mesh::global_sdr_loc[NUM_MESH_ATTR] = { 0, 1, 2, 3, 4, 5 };
   21.15 +unsigned int Mesh::intersect_mode = ISECT_DEFAULT;
   21.16 +float Mesh::vertex_sel_dist = 0.01;
   21.17 +float Mesh::vis_vecsize = 1.0;
   21.18 +
   21.19 +Mesh::Mesh()
   21.20 +{
   21.21 +	clear();
   21.22 +
   21.23 +	glGenBuffers(NUM_MESH_ATTR + 1, buffer_objects);
   21.24 +
   21.25 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
   21.26 +		vattr[i].vbo = buffer_objects[i];
   21.27 +	}
   21.28 +	ibo = buffer_objects[NUM_MESH_ATTR];
   21.29 +	wire_ibo = 0;
   21.30 +}
   21.31 +
   21.32 +Mesh::~Mesh()
   21.33 +{
   21.34 +	glDeleteBuffers(NUM_MESH_ATTR + 1, buffer_objects);
   21.35 +
   21.36 +	if(wire_ibo) {
   21.37 +		glDeleteBuffers(1, &wire_ibo);
   21.38 +	}
   21.39 +}
   21.40 +
   21.41 +void Mesh::set_name(const char *name)
   21.42 +{
   21.43 +	this->name = name;
   21.44 +}
   21.45 +
   21.46 +const char *Mesh::get_name() const
   21.47 +{
   21.48 +	return name.c_str();
   21.49 +}
   21.50 +
   21.51 +bool Mesh::has_attrib(int attr) const
   21.52 +{
   21.53 +	if(attr < 0 || attr >= NUM_MESH_ATTR) {
   21.54 +		return false;
   21.55 +	}
   21.56 +
   21.57 +	// if neither of these is valid, then nobody has set this attribute
   21.58 +	return vattr[attr].vbo_valid || vattr[attr].data_valid;
   21.59 +}
   21.60 +
   21.61 +void Mesh::clear()
   21.62 +{
   21.63 +	bones.clear();
   21.64 +
   21.65 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
   21.66 +		vattr[i].nelem = 0;
   21.67 +		vattr[i].vbo_valid = false;
   21.68 +		vattr[i].data_valid = false;
   21.69 +		//vattr[i].sdr_loc = -1;
   21.70 +		vattr[i].data.clear();
   21.71 +	}
   21.72 +	ibo_valid = idata_valid = false;
   21.73 +	idata.clear();
   21.74 +
   21.75 +	wire_ibo_valid = false;
   21.76 +
   21.77 +	nverts = nfaces = 0;
   21.78 +
   21.79 +	bsph_valid = false;
   21.80 +	aabb_valid = false;
   21.81 +}
   21.82 +
   21.83 +float *Mesh::set_attrib_data(int attrib, int nelem, unsigned int num, const float *data)
   21.84 +{
   21.85 +	if(attrib < 0 || attrib >= NUM_MESH_ATTR) {
   21.86 +		error_log("%s: invalid attrib: %d\n", __FUNCTION__, attrib);
   21.87 +		return 0;
   21.88 +	}
   21.89 +
   21.90 +	if(nverts && num != nverts) {
   21.91 +		error_log("%s: attribute count missmatch (%d instead of %d)\n", __FUNCTION__, num, nverts);
   21.92 +		return 0;
   21.93 +	}
   21.94 +	nverts = num;
   21.95 +
   21.96 +	vattr[attrib].data.clear();
   21.97 +	vattr[attrib].nelem = nelem;
   21.98 +	vattr[attrib].data.resize(num * nelem);
   21.99 +
  21.100 +	if(data) {
  21.101 +		memcpy(&vattr[attrib].data[0], data, num * nelem * sizeof *data);
  21.102 +	}
  21.103 +
  21.104 +	vattr[attrib].data_valid = true;
  21.105 +	vattr[attrib].vbo_valid = false;
  21.106 +	return &vattr[attrib].data[0];
  21.107 +}
  21.108 +
  21.109 +float *Mesh::get_attrib_data(int attrib)
  21.110 +{
  21.111 +	if(attrib < 0 || attrib >= NUM_MESH_ATTR) {
  21.112 +		error_log("%s: invalid attrib: %d\n", __FUNCTION__, attrib);
  21.113 +		return 0;
  21.114 +	}
  21.115 +
  21.116 +	vattr[attrib].vbo_valid = false;
  21.117 +	return (float*)((const Mesh*)this)->get_attrib_data(attrib);
  21.118 +}
  21.119 +
  21.120 +const float *Mesh::get_attrib_data(int attrib) const
  21.121 +{
  21.122 +	if(attrib < 0 || attrib >= NUM_MESH_ATTR) {
  21.123 +		error_log("%s: invalid attrib: %d\n", __FUNCTION__, attrib);
  21.124 +		return 0;
  21.125 +	}
  21.126 +
  21.127 +	if(!vattr[attrib].data_valid) {
  21.128 +#if GL_ES_VERSION_2_0
  21.129 +		error_log("%s: can't read back attrib data on CrippledGL ES\n", __FUNCTION__);
  21.130 +		return 0;
  21.131 +#else
  21.132 +		if(!vattr[attrib].vbo_valid) {
  21.133 +			error_log("%s: unavailable attrib: %d\n", __FUNCTION__, attrib);
  21.134 +			return 0;
  21.135 +		}
  21.136 +
  21.137 +		// local data copy is unavailable, grab the data from the vbo
  21.138 +		Mesh *m = (Mesh*)this;
  21.139 +		m->vattr[attrib].data.resize(nverts * vattr[attrib].nelem);
  21.140 +
  21.141 +		glBindBuffer(GL_ARRAY_BUFFER, vattr[attrib].vbo);
  21.142 +		void *data = glMapBuffer(GL_ARRAY_BUFFER, GL_READ_ONLY);
  21.143 +		memcpy(&m->vattr[attrib].data[0], data, nverts * vattr[attrib].nelem * sizeof(float));
  21.144 +		glUnmapBuffer(GL_ARRAY_BUFFER);
  21.145 +
  21.146 +		vattr[attrib].data_valid = true;
  21.147 +#endif
  21.148 +	}
  21.149 +
  21.150 +	return &vattr[attrib].data[0];
  21.151 +}
  21.152 +
  21.153 +void Mesh::set_attrib(int attrib, int idx, const Vector4 &v)
  21.154 +{
  21.155 +	float *data = get_attrib_data(attrib);
  21.156 +	if(data) {
  21.157 +		data += idx * vattr[attrib].nelem;
  21.158 +		for(int i=0; i<vattr[attrib].nelem; i++) {
  21.159 +			data[i] = v[i];
  21.160 +		}
  21.161 +	}
  21.162 +}
  21.163 +
  21.164 +Vector4 Mesh::get_attrib(int attrib, int idx) const
  21.165 +{
  21.166 +	Vector4 v(0.0, 0.0, 0.0, 1.0);
  21.167 +	const float *data = get_attrib_data(attrib);
  21.168 +	if(data) {
  21.169 +		data += idx * vattr[attrib].nelem;
  21.170 +		for(int i=0; i<vattr[attrib].nelem; i++) {
  21.171 +			v[i] = data[i];
  21.172 +		}
  21.173 +	}
  21.174 +	return v;
  21.175 +}
  21.176 +
  21.177 +unsigned int *Mesh::set_index_data(int num, const unsigned int *indices)
  21.178 +{
  21.179 +	int nidx = nfaces * 3;
  21.180 +	if(nidx && num != nidx) {
  21.181 +		error_log("%s: index count missmatch (%d instead of %d)\n", __FUNCTION__, num, nidx);
  21.182 +		return 0;
  21.183 +	}
  21.184 +	nfaces = num / 3;
  21.185 +
  21.186 +	idata.clear();
  21.187 +	idata.resize(num);
  21.188 +
  21.189 +	if(indices) {
  21.190 +		memcpy(&idata[0], indices, num * sizeof *indices);
  21.191 +	}
  21.192 +
  21.193 +	idata_valid = true;
  21.194 +	ibo_valid = false;
  21.195 +
  21.196 +	return &idata[0];
  21.197 +}
  21.198 +
  21.199 +unsigned int *Mesh::get_index_data()
  21.200 +{
  21.201 +	ibo_valid = false;
  21.202 +	return (unsigned int*)((const Mesh*)this)->get_index_data();
  21.203 +}
  21.204 +
  21.205 +const unsigned int *Mesh::get_index_data() const
  21.206 +{
  21.207 +	if(!idata_valid) {
  21.208 +#if GL_ES_VERSION_2_0
  21.209 +		error_log("%s: can't read back index data in CrippledGL ES\n", __FUNCTION__);
  21.210 +		return 0;
  21.211 +#else
  21.212 +		if(!ibo_valid) {
  21.213 +			error_log("%s: indices unavailable\n", __FUNCTION__);
  21.214 +			return 0;
  21.215 +		}
  21.216 +
  21.217 +		// local data copy is unavailable, gram the data from the ibo
  21.218 +		Mesh *m = (Mesh*)this;
  21.219 +		int nidx = nfaces * 3;
  21.220 +		m->idata.resize(nidx);
  21.221 +
  21.222 +		glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ibo);
  21.223 +		void *data = glMapBuffer(GL_ELEMENT_ARRAY_BUFFER, GL_READ_ONLY);
  21.224 +		memcpy(&m->idata[0], data, nidx * sizeof(unsigned int));
  21.225 +		glUnmapBuffer(GL_ELEMENT_ARRAY_BUFFER);
  21.226 +
  21.227 +		idata_valid = true;
  21.228 +#endif
  21.229 +	}
  21.230 +
  21.231 +	return &idata[0];
  21.232 +}
  21.233 +
  21.234 +void Mesh::append(const Mesh &mesh)
  21.235 +{
  21.236 +	unsigned int idxoffs = nverts;
  21.237 +
  21.238 +	nverts += mesh.nverts;
  21.239 +	nfaces += mesh.nfaces;
  21.240 +
  21.241 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.242 +		if(has_attrib(i) && mesh.has_attrib(i)) {
  21.243 +			// force validating the data arrays
  21.244 +			get_attrib_data(i);
  21.245 +			mesh.get_attrib_data(i);
  21.246 +
  21.247 +			// append the mesh data
  21.248 +			vattr[i].data.insert(vattr[i].data.end(), mesh.vattr[i].data.begin(), mesh.vattr[i].data.end());
  21.249 +		}
  21.250 +	}
  21.251 +
  21.252 +	if(ibo_valid || idata_valid) {
  21.253 +		// make index arrays valid
  21.254 +		get_index_data();
  21.255 +		mesh.get_index_data();
  21.256 +
  21.257 +		size_t orig_sz = idata.size();
  21.258 +
  21.259 +		idata.insert(idata.end(), mesh.idata.begin(), mesh.idata.end());
  21.260 +
  21.261 +		// fixup all the new indices
  21.262 +		for(size_t i=orig_sz; i<idata.size(); i++) {
  21.263 +			idata[i] += idxoffs;
  21.264 +		}
  21.265 +	}
  21.266 +
  21.267 +	// fuck everything
  21.268 +	wire_ibo_valid = false;
  21.269 +	aabb_valid = false;
  21.270 +	bsph_valid = false;
  21.271 +}
  21.272 +
  21.273 +// assemble a complete vertex by adding all the useful attributes
  21.274 +void Mesh::vertex(float x, float y, float z)
  21.275 +{
  21.276 +	cur_val[MESH_ATTR_VERTEX] = Vector4(x, y, z, 1.0f);
  21.277 +	vattr[MESH_ATTR_VERTEX].data_valid = true;
  21.278 +	vattr[MESH_ATTR_VERTEX].nelem = 3;
  21.279 +
  21.280 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.281 +		if(vattr[i].data_valid) {
  21.282 +			for(int j=0; j<vattr[MESH_ATTR_VERTEX].nelem; j++) {
  21.283 +				vattr[i].data.push_back(cur_val[i][j]);
  21.284 +			}
  21.285 +		}
  21.286 +		vattr[i].vbo_valid = false;
  21.287 +	}
  21.288 +
  21.289 +	if(idata_valid) {
  21.290 +		idata.clear();
  21.291 +	}
  21.292 +	ibo_valid = idata_valid = false;
  21.293 +}
  21.294 +
  21.295 +void Mesh::normal(float nx, float ny, float nz)
  21.296 +{
  21.297 +	cur_val[MESH_ATTR_NORMAL] = Vector4(nx, ny, nz, 1.0f);
  21.298 +	vattr[MESH_ATTR_NORMAL].data_valid = true;
  21.299 +	vattr[MESH_ATTR_NORMAL].nelem = 3;
  21.300 +}
  21.301 +
  21.302 +void Mesh::tangent(float tx, float ty, float tz)
  21.303 +{
  21.304 +	cur_val[MESH_ATTR_TANGENT] = Vector4(tx, ty, tz, 1.0f);
  21.305 +	vattr[MESH_ATTR_TANGENT].data_valid = true;
  21.306 +	vattr[MESH_ATTR_TANGENT].nelem = 3;
  21.307 +}
  21.308 +
  21.309 +void Mesh::texcoord(float u, float v, float w)
  21.310 +{
  21.311 +	cur_val[MESH_ATTR_TEXCOORD] = Vector4(u, v, w, 1.0f);
  21.312 +	vattr[MESH_ATTR_TEXCOORD].data_valid = true;
  21.313 +	vattr[MESH_ATTR_TEXCOORD].nelem = 3;
  21.314 +}
  21.315 +
  21.316 +void Mesh::boneweights(float w1, float w2, float w3, float w4)
  21.317 +{
  21.318 +	cur_val[MESH_ATTR_BONEWEIGHTS] = Vector4(w1, w2, w3, w4);
  21.319 +	vattr[MESH_ATTR_BONEWEIGHTS].data_valid = true;
  21.320 +	vattr[MESH_ATTR_BONEWEIGHTS].nelem = 4;
  21.321 +}
  21.322 +
  21.323 +void Mesh::boneidx(int idx1, int idx2, int idx3, int idx4)
  21.324 +{
  21.325 +	cur_val[MESH_ATTR_BONEIDX] = Vector4(idx1, idx2, idx3, idx4);
  21.326 +	vattr[MESH_ATTR_BONEIDX].data_valid = true;
  21.327 +	vattr[MESH_ATTR_BONEIDX].nelem = 4;
  21.328 +}
  21.329 +
  21.330 +/// static function
  21.331 +void Mesh::set_attrib_location(int attr, int loc)
  21.332 +{
  21.333 +	if(attr < 0 || attr >= NUM_MESH_ATTR) {
  21.334 +		return;
  21.335 +	}
  21.336 +	Mesh::global_sdr_loc[attr] = loc;
  21.337 +}
  21.338 +
  21.339 +/// static function
  21.340 +int Mesh::get_attrib_location(int attr)
  21.341 +{
  21.342 +	if(attr < 0 || attr >= NUM_MESH_ATTR) {
  21.343 +		return -1;
  21.344 +	}
  21.345 +	return Mesh::global_sdr_loc[attr];
  21.346 +}
  21.347 +
  21.348 +/// static function
  21.349 +void Mesh::clear_attrib_locations()
  21.350 +{
  21.351 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.352 +		Mesh::global_sdr_loc[i] = -1;
  21.353 +	}
  21.354 +}
  21.355 +
  21.356 +/// static function
  21.357 +void Mesh::set_vis_vecsize(float sz)
  21.358 +{
  21.359 +	Mesh::vis_vecsize = sz;
  21.360 +}
  21.361 +
  21.362 +float Mesh::get_vis_vecsize()
  21.363 +{
  21.364 +	return Mesh::vis_vecsize;
  21.365 +}
  21.366 +
  21.367 +void Mesh::apply_xform(const Matrix4x4 &xform)
  21.368 +{
  21.369 +	Matrix4x4 dir_xform = xform;
  21.370 +	dir_xform[0][3] = dir_xform[1][3] = dir_xform[2][3] = 0.0f;
  21.371 +	dir_xform[3][0] = dir_xform[3][1] = dir_xform[3][2] = 0.0f;
  21.372 +	dir_xform[3][3] = 1.0f;
  21.373 +
  21.374 +	apply_xform(xform, dir_xform);
  21.375 +}
  21.376 +
  21.377 +void Mesh::apply_xform(const Matrix4x4 &xform, const Matrix4x4 &dir_xform)
  21.378 +{
  21.379 +	for(unsigned int i=0; i<nverts; i++) {
  21.380 +		Vector4 v = get_attrib(MESH_ATTR_VERTEX, i);
  21.381 +		set_attrib(MESH_ATTR_VERTEX, i, v.transformed(xform));
  21.382 +
  21.383 +		if(has_attrib(MESH_ATTR_NORMAL)) {
  21.384 +			Vector3 n = get_attrib(MESH_ATTR_NORMAL, i);
  21.385 +			set_attrib(MESH_ATTR_NORMAL, i, n.transformed(dir_xform));
  21.386 +		}
  21.387 +		if(has_attrib(MESH_ATTR_TANGENT)) {
  21.388 +			Vector3 t = get_attrib(MESH_ATTR_TANGENT, i);
  21.389 +			set_attrib(MESH_ATTR_TANGENT, i, t.transformed(dir_xform));
  21.390 +		}
  21.391 +	}
  21.392 +}
  21.393 +
  21.394 +int Mesh::add_bone(XFormNode *bone)
  21.395 +{
  21.396 +	int idx = bones.size();
  21.397 +	bones.push_back(bone);
  21.398 +	return idx;
  21.399 +}
  21.400 +
  21.401 +const XFormNode *Mesh::get_bone(int idx) const
  21.402 +{
  21.403 +	if(idx < 0 || idx >= (int)bones.size()) {
  21.404 +		return 0;
  21.405 +	}
  21.406 +	return bones[idx];
  21.407 +}
  21.408 +
  21.409 +int Mesh::get_bones_count() const
  21.410 +{
  21.411 +	return (int)bones.size();
  21.412 +}
  21.413 +
  21.414 +void Mesh::draw() const
  21.415 +{
  21.416 +	const ShaderProg *cur_sdr = get_current_shader();
  21.417 +#ifdef GL_ES_VERSION_2_0
  21.418 +	if(!cur_sdr) {
  21.419 +		error_log("%s: CrippledGL ES can't draw without a shader\n", __FUNCTION__);
  21.420 +		return;
  21.421 +	}
  21.422 +#endif
  21.423 +
  21.424 +	((Mesh*)this)->update_buffers();
  21.425 +
  21.426 +	if(!vattr[MESH_ATTR_VERTEX].vbo_valid) {
  21.427 +		error_log("%s: invalid vertex buffer\n", __FUNCTION__);
  21.428 +		return;
  21.429 +	}
  21.430 +
  21.431 +	if(cur_sdr) {
  21.432 +		// rendering with shaders
  21.433 +		if(global_sdr_loc[MESH_ATTR_VERTEX] == -1) {
  21.434 +			error_log("%s: shader attribute location for vertices unset\n", __FUNCTION__);
  21.435 +			return;
  21.436 +		}
  21.437 +
  21.438 +		for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.439 +			int loc = global_sdr_loc[i];
  21.440 +			if(loc >= 0 && vattr[i].vbo_valid) {
  21.441 +				glBindBuffer(GL_ARRAY_BUFFER, vattr[i].vbo);
  21.442 +				glVertexAttribPointer(loc, vattr[i].nelem, GL_FLOAT, GL_FALSE, 0, 0);
  21.443 +				glEnableVertexAttribArray(loc);
  21.444 +			}
  21.445 +		}
  21.446 +	} else {
  21.447 +#ifndef GL_ES_VERSION_2_0
  21.448 +		// rendering with fixed-function (not available in GLES2)
  21.449 +		glBindBuffer(GL_ARRAY_BUFFER, vattr[MESH_ATTR_VERTEX].vbo);
  21.450 +		glVertexPointer(vattr[MESH_ATTR_VERTEX].nelem, GL_FLOAT, 0, 0);
  21.451 +		glEnableClientState(GL_VERTEX_ARRAY);
  21.452 +
  21.453 +		if(vattr[MESH_ATTR_NORMAL].vbo_valid) {
  21.454 +			glBindBuffer(GL_ARRAY_BUFFER, vattr[MESH_ATTR_NORMAL].vbo);
  21.455 +			glNormalPointer(GL_FLOAT, 0, 0);
  21.456 +			glEnableClientState(GL_NORMAL_ARRAY);
  21.457 +		}
  21.458 +		if(vattr[MESH_ATTR_TEXCOORD].vbo_valid) {
  21.459 +			glBindBuffer(GL_ARRAY_BUFFER, vattr[MESH_ATTR_TEXCOORD].vbo);
  21.460 +			glTexCoordPointer(vattr[MESH_ATTR_TEXCOORD].nelem, GL_FLOAT, 0, 0);
  21.461 +			glEnableClientState(GL_TEXTURE_COORD_ARRAY);
  21.462 +		}
  21.463 +		if(vattr[MESH_ATTR_COLOR].vbo_valid) {
  21.464 +			glBindBuffer(GL_ARRAY_BUFFER, vattr[MESH_ATTR_COLOR].vbo);
  21.465 +			glColorPointer(vattr[MESH_ATTR_COLOR].nelem, GL_FLOAT, 0, 0);
  21.466 +			glEnableClientState(GL_COLOR_ARRAY);
  21.467 +		}
  21.468 +#endif
  21.469 +	}
  21.470 +	glBindBuffer(GL_ARRAY_BUFFER, 0);
  21.471 +
  21.472 +	if(ibo_valid) {
  21.473 +		glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ibo);
  21.474 +		glDrawElements(GL_TRIANGLES, nfaces * 3, GL_UNSIGNED_INT, 0);
  21.475 +		glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
  21.476 +	} else {
  21.477 +		glDrawArrays(GL_TRIANGLES, 0, nverts);
  21.478 +	}
  21.479 +
  21.480 +	if(cur_sdr) {
  21.481 +		// rendered with shaders
  21.482 +		for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.483 +			int loc = global_sdr_loc[i];
  21.484 +			if(loc >= 0 && vattr[i].vbo_valid) {
  21.485 +				glDisableVertexAttribArray(loc);
  21.486 +			}
  21.487 +		}
  21.488 +	} else {
  21.489 +#ifndef GL_ES_VERSION_2_0
  21.490 +		// rendered with fixed-function
  21.491 +		glDisableClientState(GL_VERTEX_ARRAY);
  21.492 +		if(vattr[MESH_ATTR_NORMAL].vbo_valid) {
  21.493 +			glDisableClientState(GL_NORMAL_ARRAY);
  21.494 +		}
  21.495 +		if(vattr[MESH_ATTR_TEXCOORD].vbo_valid) {
  21.496 +			glDisableClientState(GL_TEXTURE_COORD_ARRAY);
  21.497 +		}
  21.498 +		if(vattr[MESH_ATTR_COLOR].vbo_valid) {
  21.499 +			glDisableClientState(GL_COLOR_ARRAY);
  21.500 +		}
  21.501 +#endif
  21.502 +	}
  21.503 +}
  21.504 +
  21.505 +void Mesh::draw_wire() const
  21.506 +{
  21.507 +	((Mesh*)this)->update_wire_ibo();
  21.508 +
  21.509 +	if(!vattr[MESH_ATTR_VERTEX].vbo_valid || !wire_ibo_valid) {
  21.510 +		error_log("%s: invalid vertex buffer\n", __FUNCTION__);
  21.511 +		return;
  21.512 +	}
  21.513 +	if(global_sdr_loc[MESH_ATTR_VERTEX] == -1) {
  21.514 +		error_log("%s: shader attribute location for vertices unset\n", __FUNCTION__);
  21.515 +		return;
  21.516 +	}
  21.517 +
  21.518 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.519 +		int loc = global_sdr_loc[i];
  21.520 +		if(loc >= 0 && vattr[i].vbo_valid) {
  21.521 +			glBindBuffer(GL_ARRAY_BUFFER, vattr[i].vbo);
  21.522 +			glVertexAttribPointer(loc, vattr[i].nelem, GL_FLOAT, GL_FALSE, 0, 0);
  21.523 +			glEnableVertexAttribArray(loc);
  21.524 +		}
  21.525 +	}
  21.526 +	glBindBuffer(GL_ARRAY_BUFFER, 0);
  21.527 +
  21.528 +	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, wire_ibo);
  21.529 +	glDrawElements(GL_LINES, nfaces * 6, GL_UNSIGNED_INT, 0);
  21.530 +	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
  21.531 +
  21.532 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.533 +		int loc = global_sdr_loc[i];
  21.534 +		if(loc >= 0 && vattr[i].vbo_valid) {
  21.535 +			glDisableVertexAttribArray(loc);
  21.536 +		}
  21.537 +	}
  21.538 +}
  21.539 +
  21.540 +void Mesh::draw_vertices() const
  21.541 +{
  21.542 +	((Mesh*)this)->update_buffers();
  21.543 +
  21.544 +	if(!vattr[MESH_ATTR_VERTEX].vbo_valid) {
  21.545 +		error_log("%s: invalid vertex buffer\n", __FUNCTION__);
  21.546 +		return;
  21.547 +	}
  21.548 +	if(global_sdr_loc[MESH_ATTR_VERTEX] == -1) {
  21.549 +		error_log("%s: shader attribute location for vertices unset\n", __FUNCTION__);
  21.550 +		return;
  21.551 +	}
  21.552 +
  21.553 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.554 +		int loc = global_sdr_loc[i];
  21.555 +		if(loc >= 0 && vattr[i].vbo_valid) {
  21.556 +			glBindBuffer(GL_ARRAY_BUFFER, vattr[i].vbo);
  21.557 +			glVertexAttribPointer(loc, vattr[i].nelem, GL_FLOAT, GL_FALSE, 0, 0);
  21.558 +			glEnableVertexAttribArray(loc);
  21.559 +		}
  21.560 +	}
  21.561 +	glBindBuffer(GL_ARRAY_BUFFER, 0);
  21.562 +
  21.563 +	glDrawArrays(GL_POINTS, 0, nverts);
  21.564 +
  21.565 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.566 +		int loc = global_sdr_loc[i];
  21.567 +		if(loc >= 0 && vattr[i].vbo_valid) {
  21.568 +			glDisableVertexAttribArray(loc);
  21.569 +		}
  21.570 +	}
  21.571 +}
  21.572 +
  21.573 +void Mesh::draw_normals() const
  21.574 +{
  21.575 +#ifdef USE_OLDGL
  21.576 +	int vert_loc = global_sdr_loc[MESH_ATTR_VERTEX];
  21.577 +	Vector3 *varr = (Vector3*)get_attrib_data(MESH_ATTR_VERTEX);
  21.578 +	Vector3 *norm = (Vector3*)get_attrib_data(MESH_ATTR_NORMAL);
  21.579 +
  21.580 +	if(!varr || !norm || vert_loc < 0) {
  21.581 +		return;
  21.582 +	}
  21.583 +
  21.584 +	glBegin(GL_LINES);
  21.585 +	for(size_t i=0; i<nverts; i++) {
  21.586 +		glVertexAttrib3f(vert_loc, varr[i].x, varr[i].y, varr[i].z);
  21.587 +		Vector3 end = varr[i] + norm[i] * vis_vecsize;
  21.588 +		glVertexAttrib3f(vert_loc, end.x, end.y, end.z);
  21.589 +	}
  21.590 +	glEnd();
  21.591 +
  21.592 +#endif	// USE_OLDGL
  21.593 +}
  21.594 +
  21.595 +void Mesh::draw_tangents() const
  21.596 +{
  21.597 +#ifdef USE_OLDGL
  21.598 +	int vert_loc = global_sdr_loc[MESH_ATTR_VERTEX];
  21.599 +	Vector3 *varr = (Vector3*)get_attrib_data(MESH_ATTR_VERTEX);
  21.600 +	Vector3 *tang = (Vector3*)get_attrib_data(MESH_ATTR_TANGENT);
  21.601 +
  21.602 +	if(!varr || !tang || vert_loc < 0) {
  21.603 +		return;
  21.604 +	}
  21.605 +
  21.606 +	glBegin(GL_LINES);
  21.607 +	for(size_t i=0; i<nverts; i++) {
  21.608 +		glVertexAttrib3f(vert_loc, varr[i].x, varr[i].y, varr[i].z);
  21.609 +		Vector3 end = varr[i] + tang[i] * vis_vecsize;
  21.610 +		glVertexAttrib3f(vert_loc, end.x, end.y, end.z);
  21.611 +	}
  21.612 +	glEnd();
  21.613 +
  21.614 +#endif	// USE_OLDGL
  21.615 +}
  21.616 +
  21.617 +void Mesh::get_aabbox(Vector3 *vmin, Vector3 *vmax) const
  21.618 +{
  21.619 +	if(!aabb_valid) {
  21.620 +		((Mesh*)this)->calc_aabb();
  21.621 +	}
  21.622 +	*vmin = aabb.min;
  21.623 +	*vmax = aabb.max;
  21.624 +}
  21.625 +
  21.626 +const AABox &Mesh::get_aabbox() const
  21.627 +{
  21.628 +	if(!aabb_valid) {
  21.629 +		((Mesh*)this)->calc_aabb();
  21.630 +	}
  21.631 +	return aabb;
  21.632 +}
  21.633 +
  21.634 +float Mesh::get_bsphere(Vector3 *center, float *rad) const
  21.635 +{
  21.636 +	if(!bsph_valid) {
  21.637 +		((Mesh*)this)->calc_bsph();
  21.638 +	}
  21.639 +	*center = bsph.center;
  21.640 +	*rad = bsph.radius;
  21.641 +	return bsph.radius;
  21.642 +}
  21.643 +
  21.644 +const Sphere &Mesh::get_bsphere() const
  21.645 +{
  21.646 +	if(!bsph_valid) {
  21.647 +		((Mesh*)this)->calc_bsph();
  21.648 +	}
  21.649 +	return bsph;
  21.650 +}
  21.651 +
  21.652 +/// static function
  21.653 +void Mesh::set_intersect_mode(unsigned int mode)
  21.654 +{
  21.655 +	Mesh::intersect_mode = mode;
  21.656 +}
  21.657 +
  21.658 +/// static function
  21.659 +unsigned int Mesh::get_intersect_mode()
  21.660 +{
  21.661 +	return Mesh::intersect_mode;
  21.662 +}
  21.663 +
  21.664 +/// static function
  21.665 +void Mesh::set_vertex_select_distance(float dist)
  21.666 +{
  21.667 +	Mesh::vertex_sel_dist = dist;
  21.668 +}
  21.669 +
  21.670 +/// static function
  21.671 +float Mesh::get_vertex_select_distance()
  21.672 +{
  21.673 +	return Mesh::vertex_sel_dist;
  21.674 +}
  21.675 +
  21.676 +bool Mesh::intersect(const Ray &ray, HitPoint *hit) const
  21.677 +{
  21.678 +	assert((Mesh::intersect_mode & (ISECT_VERTICES | ISECT_FACE)) != (ISECT_VERTICES | ISECT_FACE));
  21.679 +
  21.680 +	const Vector3 *varr = (Vector3*)get_attrib_data(MESH_ATTR_VERTEX);
  21.681 +	const Vector3 *narr = (Vector3*)get_attrib_data(MESH_ATTR_NORMAL);
  21.682 +	if(!varr) {
  21.683 +		return false;
  21.684 +	}
  21.685 +	const unsigned int *idxarr = get_index_data();
  21.686 +
  21.687 +	// first test with the bounding box
  21.688 +	AABox box;
  21.689 +	get_aabbox(&box.min, &box.max);
  21.690 +	if(!box.intersect(ray)) {
  21.691 +		return false;
  21.692 +	}
  21.693 +
  21.694 +	HitPoint nearest_hit;
  21.695 +	nearest_hit.dist = FLT_MAX;
  21.696 +	nearest_hit.obj = 0;
  21.697 +
  21.698 +	if(Mesh::intersect_mode & ISECT_VERTICES) {
  21.699 +		// we asked for "intersections" with the vertices of the mesh
  21.700 +		long nearest_vidx = -1;
  21.701 +		float thres_sq = Mesh::vertex_sel_dist * Mesh::vertex_sel_dist;
  21.702 +
  21.703 +		for(unsigned int i=0; i<nverts; i++) {
  21.704 +
  21.705 +			if((Mesh::intersect_mode & ISECT_FRONT) && dot_product(narr[i], ray.dir) > 0) {
  21.706 +				continue;
  21.707 +			}
  21.708 +
  21.709 +			// project the vertex onto the ray line
  21.710 +			float t = dot_product(varr[i] - ray.origin, ray.dir);
  21.711 +			Vector3 vproj = ray.origin + ray.dir * t;
  21.712 +
  21.713 +			float dist_sq = (vproj - varr[i]).length_sq();
  21.714 +			if(dist_sq < thres_sq) {
  21.715 +				if(!hit) {
  21.716 +					return true;
  21.717 +				}
  21.718 +				if(t < nearest_hit.dist) {
  21.719 +					nearest_hit.dist = t;
  21.720 +					nearest_vidx = i;
  21.721 +				}
  21.722 +			}
  21.723 +		}
  21.724 +
  21.725 +		if(nearest_vidx != -1) {
  21.726 +			hitvert = varr[nearest_vidx];
  21.727 +			nearest_hit.obj = &hitvert;
  21.728 +		}
  21.729 +
  21.730 +	} else {
  21.731 +		// regular intersection test with polygons
  21.732 +
  21.733 +		for(unsigned int i=0; i<nfaces; i++) {
  21.734 +			Triangle face(i, varr, idxarr);
  21.735 +
  21.736 +			// ignore back-facing polygons if the mode flags include ISECT_FRONT
  21.737 +			if((Mesh::intersect_mode & ISECT_FRONT) && dot_product(face.get_normal(), ray.dir) > 0) {
  21.738 +				continue;
  21.739 +			}
  21.740 +
  21.741 +			HitPoint fhit;
  21.742 +			if(face.intersect(ray, hit ? &fhit : 0)) {
  21.743 +				if(!hit) {
  21.744 +					return true;
  21.745 +				}
  21.746 +				if(fhit.dist < nearest_hit.dist) {
  21.747 +					nearest_hit = fhit;
  21.748 +					hitface = face;
  21.749 +				}
  21.750 +			}
  21.751 +		}
  21.752 +	}
  21.753 +
  21.754 +	if(nearest_hit.obj) {
  21.755 +		if(hit) {
  21.756 +			*hit = nearest_hit;
  21.757 +
  21.758 +			// if we are interested in the mesh and not the faces set obj to this
  21.759 +			if(Mesh::intersect_mode & ISECT_FACE) {
  21.760 +				hit->obj = &hitface;
  21.761 +			} else if(Mesh::intersect_mode & ISECT_VERTICES) {
  21.762 +				hit->obj = &hitvert;
  21.763 +			} else {
  21.764 +				hit->obj = this;
  21.765 +			}
  21.766 +		}
  21.767 +		return true;
  21.768 +	}
  21.769 +	return false;
  21.770 +}
  21.771 +
  21.772 +
  21.773 +// ------ private member functions ------
  21.774 +
  21.775 +void Mesh::calc_aabb()
  21.776 +{
  21.777 +	// the cast is to force calling the const version which doesn't invalidate
  21.778 +	if(!((const Mesh*)this)->get_attrib_data(MESH_ATTR_VERTEX)) {
  21.779 +		return;
  21.780 +	}
  21.781 +
  21.782 +	aabb.min = Vector3(FLT_MAX, FLT_MAX, FLT_MAX);
  21.783 +	aabb.max = -aabb.min;
  21.784 +
  21.785 +	for(unsigned int i=0; i<nverts; i++) {
  21.786 +		Vector4 v = get_attrib(MESH_ATTR_VERTEX, i);
  21.787 +		for(int j=0; j<3; j++) {
  21.788 +			if(v[j] < aabb.min[j]) {
  21.789 +				aabb.min[j] = v[j];
  21.790 +			}
  21.791 +			if(v[j] > aabb.max[j]) {
  21.792 +				aabb.max[j] = v[j];
  21.793 +			}
  21.794 +		}
  21.795 +	}
  21.796 +	aabb_valid = true;
  21.797 +}
  21.798 +
  21.799 +void Mesh::calc_bsph()
  21.800 +{
  21.801 +	// the cast is to force calling the const version which doesn't invalidate
  21.802 +	if(!((const Mesh*)this)->get_attrib_data(MESH_ATTR_VERTEX)) {
  21.803 +		return;
  21.804 +	}
  21.805 +
  21.806 +	Vector3 v;
  21.807 +	bsph.center = Vector3(0, 0, 0);
  21.808 +
  21.809 +	// first find the center
  21.810 +	for(unsigned int i=0; i<nverts; i++) {
  21.811 +		v = get_attrib(MESH_ATTR_VERTEX, i);
  21.812 +		bsph.center += v;
  21.813 +	}
  21.814 +	bsph.center /= (float)nverts;
  21.815 +
  21.816 +	bsph.radius = 0.0f;
  21.817 +	for(unsigned int i=0; i<nverts; i++) {
  21.818 +		v = get_attrib(MESH_ATTR_VERTEX, i);
  21.819 +		float dist_sq = (v - bsph.center).length_sq();
  21.820 +		if(dist_sq > bsph.radius) {
  21.821 +			bsph.radius = dist_sq;
  21.822 +		}
  21.823 +	}
  21.824 +	bsph.radius = sqrt(bsph.radius);
  21.825 +
  21.826 +	bsph_valid = true;
  21.827 +}
  21.828 +
  21.829 +void Mesh::update_buffers()
  21.830 +{
  21.831 +	for(int i=0; i<NUM_MESH_ATTR; i++) {
  21.832 +		if(has_attrib(i) && !vattr[i].vbo_valid) {
  21.833 +			glBindBuffer(GL_ARRAY_BUFFER, vattr[i].vbo);
  21.834 +			glBufferData(GL_ARRAY_BUFFER, nverts * vattr[i].nelem * sizeof(float), &vattr[i].data[0], GL_STATIC_DRAW);
  21.835 +			vattr[i].vbo_valid = true;
  21.836 +		}
  21.837 +	}
  21.838 +	glBindBuffer(GL_ARRAY_BUFFER, 0);
  21.839 +
  21.840 +	if(idata_valid && !ibo_valid) {
  21.841 +		glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, ibo);
  21.842 +		glBufferData(GL_ELEMENT_ARRAY_BUFFER, nfaces * 3 * sizeof(unsigned int), &idata[0], GL_STATIC_DRAW);
  21.843 +		ibo_valid = true;
  21.844 +	}
  21.845 +	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
  21.846 +}
  21.847 +
  21.848 +void Mesh::update_wire_ibo()
  21.849 +{
  21.850 +	update_buffers();
  21.851 +
  21.852 +	if(wire_ibo_valid) {
  21.853 +		return;
  21.854 +	}
  21.855 +
  21.856 +	if(!wire_ibo) {
  21.857 +		glGenBuffers(1, &wire_ibo);
  21.858 +	}
  21.859 +
  21.860 +	unsigned int *wire_idxarr = new unsigned int[nfaces * 6];
  21.861 +	unsigned int *dest = wire_idxarr;
  21.862 +
  21.863 +	if(ibo_valid) {
  21.864 +		// we're dealing with an indexed mesh
  21.865 +		const unsigned int *idxarr = ((const Mesh*)this)->get_index_data();
  21.866 +
  21.867 +		for(unsigned int i=0; i<nfaces; i++) {
  21.868 +			*dest++ = idxarr[0];
  21.869 +			*dest++ = idxarr[1];
  21.870 +			*dest++ = idxarr[1];
  21.871 +			*dest++ = idxarr[2];
  21.872 +			*dest++ = idxarr[2];
  21.873 +			*dest++ = idxarr[0];
  21.874 +			idxarr += 3;
  21.875 +		}
  21.876 +	} else {
  21.877 +		// not an indexed mesh ...
  21.878 +		for(unsigned int i=0; i<nfaces; i++) {
  21.879 +			int vidx = i * 3;
  21.880 +			*dest++ = vidx;
  21.881 +			*dest++ = vidx + 1;
  21.882 +			*dest++ = vidx + 1;
  21.883 +			*dest++ = vidx + 2;
  21.884 +			*dest++ = vidx + 2;
  21.885 +			*dest++ = vidx;
  21.886 +		}
  21.887 +	}
  21.888 +
  21.889 +	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, wire_ibo);
  21.890 +	glBufferData(GL_ELEMENT_ARRAY_BUFFER, nfaces * 6 * sizeof(unsigned int), wire_idxarr, GL_STATIC_DRAW);
  21.891 +	delete [] wire_idxarr;
  21.892 +	wire_ibo_valid = true;
  21.893 +	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
  21.894 +}
  21.895 +
  21.896 +
  21.897 +// ------ class Triangle ------
  21.898 +Triangle::Triangle()
  21.899 +{
  21.900 +	normal_valid = false;
  21.901 +	id = -1;
  21.902 +}
  21.903 +
  21.904 +Triangle::Triangle(const Vector3 &v0, const Vector3 &v1, const Vector3 &v2)
  21.905 +{
  21.906 +	v[0] = v0;
  21.907 +	v[1] = v1;
  21.908 +	v[2] = v2;
  21.909 +	normal_valid = false;
  21.910 +	id = -1;
  21.911 +}
  21.912 +
  21.913 +Triangle::Triangle(int n, const Vector3 *varr, const unsigned int *idxarr)
  21.914 +{
  21.915 +	if(idxarr) {
  21.916 +		v[0] = varr[idxarr[n * 3]];
  21.917 +		v[1] = varr[idxarr[n * 3 + 1]];
  21.918 +		v[2] = varr[idxarr[n * 3 + 2]];
  21.919 +	} else {
  21.920 +		v[0] = varr[n * 3];
  21.921 +		v[1] = varr[n * 3 + 1];
  21.922 +		v[2] = varr[n * 3 + 2];
  21.923 +	}
  21.924 +	normal_valid = false;
  21.925 +	id = n;
  21.926 +}
  21.927 +
  21.928 +void Triangle::calc_normal()
  21.929 +{
  21.930 +	normal = cross_product(v[1] - v[0], v[2] - v[0]).normalized();
  21.931 +	normal_valid = true;
  21.932 +}
  21.933 +
  21.934 +const Vector3 &Triangle::get_normal() const
  21.935 +{
  21.936 +	if(!normal_valid) {
  21.937 +		((Triangle*)this)->calc_normal();
  21.938 +	}
  21.939 +	return normal;
  21.940 +}
  21.941 +
  21.942 +void Triangle::transform(const Matrix4x4 &xform)
  21.943 +{
  21.944 +	v[0].transform(xform);
  21.945 +	v[1].transform(xform);
  21.946 +	v[2].transform(xform);
  21.947 +	normal_valid = false;
  21.948 +}
  21.949 +
  21.950 +void Triangle::draw() const
  21.951 +{
  21.952 +	Vector3 n[3];
  21.953 +	n[0] = get_normal();
  21.954 +	n[1] = get_normal();
  21.955 +	n[2] = get_normal();
  21.956 +
  21.957 +	int vloc = Mesh::get_attrib_location(MESH_ATTR_VERTEX);
  21.958 +	int nloc = Mesh::get_attrib_location(MESH_ATTR_NORMAL);
  21.959 +
  21.960 +	glEnableVertexAttribArray(vloc);
  21.961 +	glVertexAttribPointer(vloc, 3, GL_FLOAT, GL_FALSE, 0, &v[0].x);
  21.962 +	glVertexAttribPointer(nloc, 3, GL_FLOAT, GL_FALSE, 0, &n[0].x);
  21.963 +
  21.964 +	glDrawArrays(GL_TRIANGLES, 0, 3);
  21.965 +
  21.966 +	glDisableVertexAttribArray(vloc);
  21.967 +	glDisableVertexAttribArray(nloc);
  21.968 +	CHECKGLERR;
  21.969 +}
  21.970 +
  21.971 +void Triangle::draw_wire() const
  21.972 +{
  21.973 +	static const int idxarr[] = {0, 1, 1, 2, 2, 0};
  21.974 +	int vloc = Mesh::get_attrib_location(MESH_ATTR_VERTEX);
  21.975 +
  21.976 +	glEnableVertexAttribArray(vloc);
  21.977 +	glVertexAttribPointer(vloc, 3, GL_FLOAT, GL_FALSE, 0, &v[0].x);
  21.978 +
  21.979 +	glDrawElements(GL_LINES, 6, GL_UNSIGNED_INT, idxarr);
  21.980 +
  21.981 +	glDisableVertexAttribArray(vloc);
  21.982 +	CHECKGLERR;
  21.983 +}
  21.984 +
  21.985 +Vector3 Triangle::calc_barycentric(const Vector3 &pos) const
  21.986 +{
  21.987 +	Vector3 norm = get_normal();
  21.988 +
  21.989 +	float area_sq = fabs(dot_product(cross_product(v[1] - v[0], v[2] - v[0]), norm));
  21.990 +	if(area_sq < 1e-5) {
  21.991 +		return Vector3(0, 0, 0);
  21.992 +	}
  21.993 +
  21.994 +	float asq0 = fabs(dot_product(cross_product(v[1] - pos, v[2] - pos), norm));
  21.995 +	float asq1 = fabs(dot_product(cross_product(v[2] - pos, v[0] - pos), norm));
  21.996 +	float asq2 = fabs(dot_product(cross_product(v[0] - pos, v[1] - pos), norm));
  21.997 +
  21.998 +	return Vector3(asq0 / area_sq, asq1 / area_sq, asq2 / area_sq);
  21.999 +}
 21.1000 +
 21.1001 +bool Triangle::intersect(const Ray &ray, HitPoint *hit) const
 21.1002 +{
 21.1003 +	Vector3 normal = get_normal();
 21.1004 +
 21.1005 +	float ndotdir = dot_product(ray.dir, normal);
 21.1006 +	if(fabs(ndotdir) < 1e-4) {
 21.1007 +		return false;
 21.1008 +	}
 21.1009 +
 21.1010 +	Vector3 vertdir = v[0] - ray.origin;
 21.1011 +	float t = dot_product(normal, vertdir) / ndotdir;
 21.1012 +
 21.1013 +	Vector3 pos = ray.origin + ray.dir * t;
 21.1014 +	Vector3 bary = calc_barycentric(pos);
 21.1015 +
 21.1016 +	if(bary.x + bary.y + bary.z > 1.00001) {
 21.1017 +		return false;
 21.1018 +	}
 21.1019 +
 21.1020 +	if(hit) {
 21.1021 +		hit->dist = t;
 21.1022 +		hit->pos = ray.origin + ray.dir * t;
 21.1023 +		hit->normal = normal;
 21.1024 +		hit->obj = this;
 21.1025 +	}
 21.1026 +	return true;
 21.1027 +}
    22.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    22.2 +++ b/src/mesh.h	Sun Aug 24 09:41:24 2014 +0300
    22.3 @@ -0,0 +1,210 @@
    22.4 +#ifndef MESH_H_
    22.5 +#define MESH_H_
    22.6 +
    22.7 +#include <string>
    22.8 +#include <vector>
    22.9 +#include <vmath/vmath.h>
   22.10 +#include "geom.h"
   22.11 +
   22.12 +enum {
   22.13 +	MESH_ATTR_VERTEX,
   22.14 +	MESH_ATTR_NORMAL,
   22.15 +	MESH_ATTR_TANGENT,
   22.16 +	MESH_ATTR_TEXCOORD,
   22.17 +	MESH_ATTR_COLOR,
   22.18 +	MESH_ATTR_BONEWEIGHTS,
   22.19 +	MESH_ATTR_BONEIDX,
   22.20 +
   22.21 +	NUM_MESH_ATTR
   22.22 +};
   22.23 +
   22.24 +// intersection mode flags
   22.25 +enum {
   22.26 +	ISECT_DEFAULT	= 0,	// default (whole mesh, all intersections)
   22.27 +	ISECT_FRONT		= 1,	// front-faces only
   22.28 +	ISECT_FACE		= 2,	// return intersected face pointer instead of mesh
   22.29 +	ISECT_VERTICES	= 4		// return (?) TODO
   22.30 +};
   22.31 +
   22.32 +class XFormNode;
   22.33 +
   22.34 +
   22.35 +class Triangle {
   22.36 +public:
   22.37 +	Vector3 v[3];
   22.38 +	Vector3 normal;
   22.39 +	bool normal_valid;
   22.40 +	int id;
   22.41 +
   22.42 +	Triangle();
   22.43 +	Triangle(const Vector3 &v0, const Vector3 &v1, const Vector3 &v2);
   22.44 +	Triangle(int n, const Vector3 *varr, const unsigned int *idxarr = 0);
   22.45 +
   22.46 +	/// calculate normal (quite expensive)
   22.47 +	void calc_normal();
   22.48 +	const Vector3 &get_normal() const;
   22.49 +
   22.50 +	void transform(const Matrix4x4 &xform);
   22.51 +
   22.52 +	void draw() const;
   22.53 +	void draw_wire() const;
   22.54 +
   22.55 +	/// calculate barycentric coordinates of a point
   22.56 +	Vector3 calc_barycentric(const Vector3 &pos) const;
   22.57 +
   22.58 +	bool intersect(const Ray &ray, HitPoint *hit = 0) const;
   22.59 +};
   22.60 +
   22.61 +
   22.62 +class Mesh {
   22.63 +private:
   22.64 +	std::string name;
   22.65 +	unsigned int nverts, nfaces;
   22.66 +
   22.67 +	// current value for each attribute for the immedate mode
   22.68 +	// interface.
   22.69 +	Vector4 cur_val[NUM_MESH_ATTR];
   22.70 +
   22.71 +	unsigned int buffer_objects[NUM_MESH_ATTR + 1];
   22.72 +
   22.73 +	// vertex attribute data and buffer objects
   22.74 +	struct {
   22.75 +		int nelem;					// number of elements per attribute range: [1, 4]
   22.76 +		std::vector<float> data;
   22.77 +		unsigned int vbo;
   22.78 +		mutable bool vbo_valid;		// if this is false, the vbo needs updating from the data
   22.79 +		mutable bool data_valid;	// if this is false, the data needs to be pulled from the vbo
   22.80 +		//int sdr_loc;
   22.81 +	} vattr[NUM_MESH_ATTR];
   22.82 +
   22.83 +	static int global_sdr_loc[NUM_MESH_ATTR];
   22.84 +
   22.85 +	std::vector<XFormNode*> bones;	// bones affecting this mesh
   22.86 +
   22.87 +	// index data and buffer object
   22.88 +	std::vector<unsigned int> idata;
   22.89 +	unsigned int ibo;
   22.90 +	mutable bool ibo_valid;
   22.91 +	mutable bool idata_valid;
   22.92 +
   22.93 +	// index buffer object for wireframe rendering (constructed on demand)
   22.94 +	unsigned int wire_ibo;
   22.95 +	mutable bool wire_ibo_valid;
   22.96 +
   22.97 +	// axis-aligned bounding box
   22.98 +	mutable AABox aabb;
   22.99 +	mutable bool aabb_valid;
  22.100 +
  22.101 +	// bounding sphere
  22.102 +	mutable Sphere bsph;
  22.103 +	mutable bool bsph_valid;
  22.104 +
  22.105 +	// keeps the last intersected face
  22.106 +	mutable Triangle hitface;
  22.107 +	// keeps the last intersected vertex position
  22.108 +	mutable Vector3 hitvert;
  22.109 +
  22.110 +	void calc_aabb();
  22.111 +	void calc_bsph();
  22.112 +
  22.113 +	static unsigned int intersect_mode;
  22.114 +	static float vertex_sel_dist;
  22.115 +
  22.116 +	static float vis_vecsize;
  22.117 +
  22.118 +	/// update the VBOs after data has changed (invalid vbo/ibo)
  22.119 +	void update_buffers();
  22.120 +	/// construct/update the wireframe index buffer (called from draw_wire).
  22.121 +	void update_wire_ibo();
  22.122 +
  22.123 +
  22.124 +public:
  22.125 +	Mesh();
  22.126 +	~Mesh();
  22.127 +
  22.128 +	void set_name(const char *name);
  22.129 +	const char *get_name() const;
  22.130 +
  22.131 +	bool has_attrib(int attr) const;
  22.132 +
  22.133 +	// clears everything about this mesh, and returns to the newly constructed state
  22.134 +	void clear();
  22.135 +
  22.136 +	// access the vertex attribute data
  22.137 +	// if vdata == 0, space is just allocated
  22.138 +	float *set_attrib_data(int attrib, int nelem, unsigned int num, const float *vdata = 0); // invalidates vbo
  22.139 +	float *get_attrib_data(int attrib);	// invalidates vbo
  22.140 +	const float *get_attrib_data(int attrib) const;
  22.141 +
  22.142 +	// simple access to any particular attribute
  22.143 +	void set_attrib(int attrib, int idx, const Vector4 &v); // invalidates vbo
  22.144 +	Vector4 get_attrib(int attrib, int idx) const;
  22.145 +
  22.146 +	// ... same for index data
  22.147 +	unsigned int *set_index_data(int num, const unsigned int *indices = 0); // invalidates ibo
  22.148 +	unsigned int *get_index_data();	// invalidates ibo
  22.149 +	const unsigned int *get_index_data() const;
  22.150 +
  22.151 +	void append(const Mesh &mesh);
  22.152 +
  22.153 +	// immediate-mode style mesh construction interface
  22.154 +	void vertex(float x, float y, float z);
  22.155 +	void normal(float nx, float ny, float nz);
  22.156 +	void tangent(float tx, float ty, float tz);
  22.157 +	void texcoord(float u, float v, float w);
  22.158 +	void boneweights(float w1, float w2, float w3, float w4);
  22.159 +	void boneidx(int idx1, int idx2, int idx3, int idx4);
  22.160 +
  22.161 +	/* apply a transformation to the vertices and its inverse-transpose
  22.162 +	 * to the normals and tangents.
  22.163 +	 */
  22.164 +	void apply_xform(const Matrix4x4 &xform);
  22.165 +	void apply_xform(const Matrix4x4 &xform, const Matrix4x4 &dir_xform);
  22.166 +
  22.167 +	// adds a bone and returns its index
  22.168 +	int add_bone(XFormNode *bone);
  22.169 +	const XFormNode *get_bone(int idx) const;
  22.170 +	int get_bones_count() const;
  22.171 +
  22.172 +	// access the shader attribute locations
  22.173 +	static void set_attrib_location(int attr, int loc);
  22.174 +	static int get_attrib_location(int attr);
  22.175 +	static void clear_attrib_locations();
  22.176 +
  22.177 +	static void set_vis_vecsize(float sz);
  22.178 +	static float get_vis_vecsize();
  22.179 +
  22.180 +	void draw() const;
  22.181 +	void draw_wire() const;
  22.182 +	void draw_vertices() const;
  22.183 +	void draw_normals() const;
  22.184 +	void draw_tangents() const;
  22.185 +
  22.186 +	/** get the bounding box in local space. The result will be cached, and subsequent
  22.187 +	 * calls will return the same box. The cache gets invalidated by any functions that can affect
  22.188 +	 * the vertex data (non-const variant of get_attrib_data(MESH_ATTR_VERTEX, ...) included).
  22.189 +	 * @{ */
  22.190 +	void get_aabbox(Vector3 *vmin, Vector3 *vmax) const;
  22.191 +	const AABox &get_aabbox() const;
  22.192 +	/// @}
  22.193 +
  22.194 +	/** get the bounding sphere in local space. The result will be cached, and subsequent
  22.195 +	 * calls will return the same box. The cache gets invalidated by any functions that can affect
  22.196 +	 * the vertex data (non-const variant of get_attrib_data(MESH_ATTR_VERTEX, ...) included).
  22.197 +	 * @{ */
  22.198 +	float get_bsphere(Vector3 *center, float *rad) const;
  22.199 +	const Sphere &get_bsphere() const;
  22.200 +
  22.201 +	static void set_intersect_mode(unsigned int mode);
  22.202 +	static unsigned int get_intersect_mode();
  22.203 +	static void set_vertex_select_distance(float dist);
  22.204 +	static float get_vertex_select_distance();
  22.205 +
  22.206 +	/** Find the intersection between the mesh and a ray.
  22.207 +	 * XXX Brute force at the moment, not intended to be used for anything other than picking in tools.
  22.208 +	 *     If you intend to use it in a speed-critical part of the code, you'll *have* to optimize it!
  22.209 +	 */
  22.210 +	bool intersect(const Ray &ray, HitPoint *hit = 0) const;
  22.211 +};
  22.212 +
  22.213 +#endif	// MESH_H_
    23.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    23.2 +++ b/src/object.cc	Sun Aug 24 09:41:24 2014 +0300
    23.3 @@ -0,0 +1,285 @@
    23.4 +#include <float.h>
    23.5 +#include "object.h"
    23.6 +#include "opengl.h"
    23.7 +#include "unistate.h"
    23.8 +#include "logger.h"
    23.9 +
   23.10 +
   23.11 +static void destroy_all_rec(XFormNode *node);
   23.12 +static void get_all_meshes_rec(XFormNode *node, std::list<Mesh*> *reslist);
   23.13 +
   23.14 +DrawMode Object::draw_mode = DRAW_DEFAULT;
   23.15 +
   23.16 +Object::Object()
   23.17 +{
   23.18 +	mesh = 0;
   23.19 +}
   23.20 +
   23.21 +void Object::destroy_all()
   23.22 +{
   23.23 +	destroy_all_rec(this);
   23.24 +}
   23.25 +
   23.26 +static void destroy_all_rec(XFormNode *node)
   23.27 +{
   23.28 +	if(!node) {
   23.29 +		return;
   23.30 +	}
   23.31 +
   23.32 +	for(int i=0; i<node->get_children_count(); i++) {
   23.33 +		destroy_all_rec(node->get_child(i));
   23.34 +	}
   23.35 +	delete node;
   23.36 +}
   23.37 +
   23.38 +void Object::set_mesh(Mesh *m)
   23.39 +{
   23.40 +	mesh = m;
   23.41 +}
   23.42 +
   23.43 +Mesh *Object::get_mesh()
   23.44 +{
   23.45 +	return mesh;
   23.46 +}
   23.47 +
   23.48 +const Mesh *Object::get_mesh() const
   23.49 +{
   23.50 +	return mesh;
   23.51 +}
   23.52 +
   23.53 +std::list<Mesh*> Object::get_all_meshes()
   23.54 +{
   23.55 +	std::list<Mesh*> meshes;
   23.56 +	get_all_meshes_rec(this, &meshes);
   23.57 +	return meshes;
   23.58 +}
   23.59 +
   23.60 +std::list<const Mesh*> Object::get_all_meshes() const
   23.61 +{
   23.62 +	std::list<const Mesh*> meshes;
   23.63 +	get_all_meshes_rec((Object*)this, (std::list<Mesh*>*)&meshes);
   23.64 +	return meshes;
   23.65 +}
   23.66 +
   23.67 +static void get_all_meshes_rec(XFormNode *node, std::list<Mesh*> *reslist)
   23.68 +{
   23.69 +	if(!node) {
   23.70 +		return;
   23.71 +	}
   23.72 +
   23.73 +	Object *obj = dynamic_cast<Object*>(node);
   23.74 +	if(obj) {
   23.75 +		Mesh *mesh = obj->get_mesh();
   23.76 +		if(mesh) {
   23.77 +			reslist->push_back(mesh);
   23.78 +		}
   23.79 +	}
   23.80 +
   23.81 +	for(int i=0; i<node->get_children_count(); i++) {
   23.82 +		get_all_meshes_rec(node->get_child(i), reslist);
   23.83 +	}
   23.84 +}
   23.85 +
   23.86 +AABox Object::get_aabbox() const
   23.87 +{
   23.88 +	AABox box;
   23.89 +
   23.90 +	if(mesh) {
   23.91 +		box = mesh->get_aabbox();
   23.92 +	} else {
   23.93 +		box.min = Vector3(FLT_MAX, FLT_MAX, FLT_MAX);
   23.94 +		box.max = -box.min;
   23.95 +	}
   23.96 +
   23.97 +	int num_children = get_children_count();
   23.98 +	for(int i=0; i<num_children; i++) {
   23.99 +		const Object *obj = dynamic_cast<const Object*>(get_child(i));
  23.100 +		if(obj) {
  23.101 +			AABox child_box = obj->get_aabbox();
  23.102 +			box.set_union(&box, &child_box);
  23.103 +		}
  23.104 +	}
  23.105 +	return box;
  23.106 +}
  23.107 +
  23.108 +Sphere Object::get_bsphere() const
  23.109 +{
  23.110 +	Sphere sph;
  23.111 +	bool valid = false;
  23.112 +
  23.113 +	if(mesh) {
  23.114 +		sph = mesh->get_bsphere();
  23.115 +		valid = true;
  23.116 +	} else {
  23.117 +		sph.radius = 0.0;
  23.118 +	}
  23.119 +
  23.120 +	int num_children = get_children_count();
  23.121 +	for(int i=0; i<num_children; i++) {
  23.122 +		const Object *obj = dynamic_cast<const Object*>(get_child(i));
  23.123 +		if(obj) {
  23.124 +			Sphere child_sph = obj->get_bsphere();
  23.125 +			if(valid) {
  23.126 +				sph.set_union(&sph, &child_sph);
  23.127 +			} else {
  23.128 +				sph = child_sph;
  23.129 +				valid = true;
  23.130 +			}
  23.131 +		}
  23.132 +	}
  23.133 +
  23.134 +	return sph;
  23.135 +}
  23.136 +
  23.137 +/*static const char *attr_name[] = {
  23.138 +	"attr_vertex",
  23.139 +	"attr_normal",
  23.140 +	"attr_tangent",
  23.141 +	"attr_texcoord",
  23.142 +	"attr_boneweights",
  23.143 +	"attr_boneidx"
  23.144 +};*/
  23.145 +
  23.146 +void Object::draw(long msec) const
  23.147 +{
  23.148 +	Matrix4x4 xform;
  23.149 +	get_xform(msec, &xform);
  23.150 +
  23.151 +	set_world_matrix(xform);
  23.152 +
  23.153 +	if(mesh) {
  23.154 +		/*unsigned int prog = sdrprog;
  23.155 +
  23.156 +		if(mesh->get_bones_count() > 0) {
  23.157 +			prog = sdrprog_skin;
  23.158 +		}
  23.159 +
  23.160 +		glUseProgram(prog);
  23.161 +		// get all the attribute locations
  23.162 +		for(int i=0; i<NUM_MESH_ATTR; i++) {
  23.163 +			int loc = glGetAttribLocation(prog, attr_name[i]);
  23.164 +			if(loc != -1) {
  23.165 +				Mesh::set_attrib_location(i, loc);
  23.166 +			}
  23.167 +		}
  23.168 +
  23.169 +		setup_bones(msec);
  23.170 +
  23.171 +		glUseProgram(prog);*/
  23.172 +
  23.173 +		material.setup();
  23.174 +		setup_unistate();	// set all state uniforms
  23.175 +
  23.176 +		switch(Object::draw_mode) {
  23.177 +		case DRAW_WIREFRAME:
  23.178 +			mesh->draw_wire();
  23.179 +			break;
  23.180 +
  23.181 +		case DRAW_VERTICES:
  23.182 +			mesh->draw_vertices();
  23.183 +			break;
  23.184 +
  23.185 +		case DRAW_DEFAULT:
  23.186 +		default:
  23.187 +			mesh->draw();
  23.188 +		}
  23.189 +	}
  23.190 +
  23.191 +	int num_children = get_children_count();
  23.192 +	for(int i=0; i<num_children; i++) {
  23.193 +		const Object *obj = dynamic_cast<const Object*>(get_child(i));
  23.194 +		if(obj) {
  23.195 +			obj->draw(msec);
  23.196 +		}
  23.197 +	}
  23.198 +}
  23.199 +
  23.200 +
  23.201 +bool Object::intersect(const Ray &inray, HitPoint *hit) const
  23.202 +{
  23.203 +	Ray ray = inray;
  23.204 +	Matrix4x4 xform, inv_xform;
  23.205 +	get_xform(ray.time, &xform/*, &inv_xform*/);
  23.206 +	ray.transform(xform.inverse());	// TODO find out what's wrong with get_xform's inv_xform and use that
  23.207 +
  23.208 +	HitPoint nearest_hit;
  23.209 +	nearest_hit.dist = FLT_MAX;
  23.210 +	nearest_hit.obj = 0;
  23.211 +
  23.212 +	if(mesh) {
  23.213 +		if(mesh->intersect(ray, hit ? &nearest_hit : 0)) {
  23.214 +			if(!hit) {
  23.215 +				return true;
  23.216 +			}
  23.217 +
  23.218 +			if(Mesh::get_intersect_mode() & ISECT_FACE) {
  23.219 +				Triangle *face = (Triangle*)nearest_hit.obj;
  23.220 +				face->transform(xform);
  23.221 +			} else if(Mesh::get_intersect_mode() & ISECT_VERTICES) {
  23.222 +				Vector3 *v = (Vector3*)nearest_hit.obj;
  23.223 +				v->transform(xform);
  23.224 +			} else {
  23.225 +				nearest_hit.obj = this;
  23.226 +			}
  23.227 +		}
  23.228 +	}
  23.229 +
  23.230 +	int num_children = get_children_count();
  23.231 +	for(int i=0; i<num_children; i++) {
  23.232 +		const Object *obj = dynamic_cast<const Object*>(get_child(i));
  23.233 +
  23.234 +		HitPoint chit;
  23.235 +		if(obj && obj->intersect(inray, hit ? &chit : 0)) {
  23.236 +			if(!hit) {
  23.237 +				return true;
  23.238 +			}
  23.239 +
  23.240 +			if(chit.dist < nearest_hit.dist) {
  23.241 +				nearest_hit = chit;
  23.242 +			}
  23.243 +		}
  23.244 +	}
  23.245 +
  23.246 +	if(nearest_hit.obj) {
  23.247 +		if(hit) {
  23.248 +			*hit = nearest_hit;
  23.249 +		}
  23.250 +		return true;
  23.251 +	}
  23.252 +	return false;
  23.253 +}
  23.254 +
  23.255 +
  23.256 +bool Object::setup_bones(long msec) const
  23.257 +{
  23.258 +	int num_bones;
  23.259 +	if(!mesh || !(num_bones = mesh->get_bones_count())) {
  23.260 +		return false;
  23.261 +	}
  23.262 +
  23.263 +	/*char uniname[32];
  23.264 +
  23.265 +	for(int i=0; i<num_bones; i++) {
  23.266 +		const XFormNode *bone = mesh->get_bone(i);
  23.267 +
  23.268 +		Matrix4x4 xform;
  23.269 +		bone->get_xform(msec, &xform);
  23.270 +
  23.271 +		xform = xform * bone->get_bone_matrix();
  23.272 +
  23.273 +		sprintf(uniname, "bone_xform[%d]", i);
  23.274 +		int loc = glGetUniformLocation(sdrprog_skin, uniname);
  23.275 +		if(loc == -1) {
  23.276 +			return false;
  23.277 +		}
  23.278 +		glUniformMatrix4fv(loc, 1, GL_TRUE, xform[0]);
  23.279 +	}*/
  23.280 +	return true;
  23.281 +}
  23.282 +
  23.283 +DrawMode Object::set_draw_mode(DrawMode mode)
  23.284 +{
  23.285 +	DrawMode prev = Object::draw_mode;
  23.286 +	Object::draw_mode = mode;
  23.287 +	return prev;
  23.288 +}
    24.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    24.2 +++ b/src/object.h	Sun Aug 24 09:41:24 2014 +0300
    24.3 @@ -0,0 +1,51 @@
    24.4 +#ifndef OBJECT_H_
    24.5 +#define OBJECT_H_
    24.6 +
    24.7 +#include <list>
    24.8 +#include "xform_node.h"
    24.9 +#include "mesh.h"
   24.10 +#include "material.h"
   24.11 +
   24.12 +enum DrawMode {
   24.13 +	DRAW_DEFAULT,
   24.14 +	DRAW_WIREFRAME,
   24.15 +	DRAW_VERTICES
   24.16 +};
   24.17 +
   24.18 +
   24.19 +class Object : public XFormNode {
   24.20 +private:
   24.21 +	Mesh *mesh;	///< no ownership, just keeping the pointer around
   24.22 +	static DrawMode draw_mode;
   24.23 +
   24.24 +	bool setup_bones(long msec) const;
   24.25 +
   24.26 +public:
   24.27 +	Material material;
   24.28 +
   24.29 +	Object();
   24.30 +
   24.31 +	/// destroy this object and all the hierarchy of objects hanging below it
   24.32 +	void destroy_all();
   24.33 +
   24.34 +	void set_mesh(Mesh *m);
   24.35 +	Mesh *get_mesh();
   24.36 +	const Mesh *get_mesh() const;
   24.37 +
   24.38 +	/// get all the meshes of the subtree rooted in this object @{
   24.39 +	std::list<Mesh*> get_all_meshes();
   24.40 +	std::list<const Mesh*> get_all_meshes() const;
   24.41 +	/// @}
   24.42 +
   24.43 +	AABox get_aabbox() const;
   24.44 +	Sphere get_bsphere() const;
   24.45 +
   24.46 +	void draw(long msec = 0) const;
   24.47 +
   24.48 +	bool intersect(const Ray &ray, HitPoint *hit = 0) const;
   24.49 +
   24.50 +	/// this is mostly for tools, to allow drawing only the wireframe or vertices
   24.51 +	static DrawMode set_draw_mode(DrawMode mode);
   24.52 +};
   24.53 +
   24.54 +#endif	// OBJECT_H_
    25.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    25.2 +++ b/src/scene.cc	Sun Aug 24 09:41:24 2014 +0300
    25.3 @@ -0,0 +1,34 @@
    25.4 +#include "scene.h"
    25.5 +#include "opengl.h"
    25.6 +
    25.7 +Scene::~Scene()
    25.8 +{
    25.9 +	destroy();
   25.10 +}
   25.11 +
   25.12 +void Scene::destroy()
   25.13 +{
   25.14 +	for(size_t i=0; i<objects.size(); i++) {
   25.15 +		delete objects[i];
   25.16 +	}
   25.17 +	objects.clear();
   25.18 +
   25.19 +	for(size_t i=0; i<meshes.size(); i++) {
   25.20 +		delete meshes[i];
   25.21 +	}
   25.22 +	meshes.clear();
   25.23 +}
   25.24 +
   25.25 +void Scene::draw(long msec) const
   25.26 +{
   25.27 +	for(size_t i=0; i<objects.size(); i++) {
   25.28 +		objects[i]->draw(msec);
   25.29 +	}
   25.30 +
   25.31 +	// if there are no objects in the scene, just draw the meshes instead
   25.32 +	if(objects.empty()) {
   25.33 +		for(size_t i=0; i<meshes.size(); i++) {
   25.34 +			meshes[i]->draw();
   25.35 +		}
   25.36 +	}
   25.37 +}
    26.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    26.2 +++ b/src/scene.h	Sun Aug 24 09:41:24 2014 +0300
    26.3 @@ -0,0 +1,25 @@
    26.4 +#ifndef SCENE_H_
    26.5 +#define SCENE_H_
    26.6 +
    26.7 +#include <vector>
    26.8 +#include "object.h"
    26.9 +#include "light.h"
   26.10 +#include "camera.h"
   26.11 +
   26.12 +class Scene {
   26.13 +public:
   26.14 +	std::vector<Object*> objects;
   26.15 +	std::vector<Mesh*> meshes;
   26.16 +	std::vector<Light*> lights;
   26.17 +	std::vector<Camera*> cameras;
   26.18 +
   26.19 +	// nodes can be objects, lights, cameras, or just dummy nodes
   26.20 +	std::vector<XFormNode*> nodes;
   26.21 +
   26.22 +	~Scene();
   26.23 +	void destroy();
   26.24 +
   26.25 +	void draw(long msec = 0) const;
   26.26 +};
   26.27 +
   26.28 +#endif	// SCENE_H_
    27.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    27.2 +++ b/src/shader.cc	Sun Aug 24 09:41:24 2014 +0300
    27.3 @@ -0,0 +1,671 @@
    27.4 +#include <stdio.h>
    27.5 +#include <string.h>
    27.6 +#include <stdarg.h>
    27.7 +#include <errno.h>
    27.8 +#include "opengl.h"
    27.9 +#include "shader.h"
   27.10 +#include "mesh.h"
   27.11 +#include "logger.h"
   27.12 +#include "unistate.h"
   27.13 +
   27.14 +#ifdef _MSC_VER
   27.15 +#include <malloc.h>
   27.16 +#else
   27.17 +#include <alloca.h>
   27.18 +#endif
   27.19 +
   27.20 +#ifdef __GLEW_H__
   27.21 +#define HAVE_GEOMETRY_SHADER
   27.22 +#define HAVE_TESSELATION_SHADER
   27.23 +#endif
   27.24 +
   27.25 +static std::string read_source(const char *fname);
   27.26 +static void bind_standard_attr(const ShaderProg *prog);
   27.27 +
   27.28 +ShaderProg *ShaderProg::current;
   27.29 +
   27.30 +void bind_shader(const ShaderProg *sdr)
   27.31 +{
   27.32 +	if(sdr) {
   27.33 +		sdr->bind();
   27.34 +	} else {
   27.35 +#ifndef GL_ES_VERSION_2_0
   27.36 +		glUseProgram(0);
   27.37 +		ShaderProg::current = 0;
   27.38 +#endif
   27.39 +	}
   27.40 +}
   27.41 +
   27.42 +const ShaderProg *get_current_shader()
   27.43 +{
   27.44 +	return ShaderProg::current;
   27.45 +}
   27.46 +
   27.47 +Shader::Shader()
   27.48 +{
   27.49 +	sdr = type = 0;
   27.50 +}
   27.51 +
   27.52 +Shader::~Shader()
   27.53 +{
   27.54 +	destroy();
   27.55 +}
   27.56 +
   27.57 +unsigned int Shader::get_id() const
   27.58 +{
   27.59 +	return sdr;
   27.60 +}
   27.61 +
   27.62 +void Shader::set_name(const char *name)
   27.63 +{
   27.64 +	this->name = std::string(name);
   27.65 +}
   27.66 +
   27.67 +const char *Shader::get_name() const
   27.68 +{
   27.69 +	return name.c_str();
   27.70 +}
   27.71 +
   27.72 +bool Shader::create(const char *src, unsigned int type)
   27.73 +{
   27.74 +#if !GL_ES_VERSION_2_0
   27.75 +	const char *src_arr[] = {src};
   27.76 +#else
   27.77 +	const char *src_arr[] = { "precision mediump float; ", src };
   27.78 +#endif
   27.79 +
   27.80 +	// keep a copy of the source code
   27.81 +	this->src = std::string(src);
   27.82 +
   27.83 +	if(!sdr) {
   27.84 +		sdr = glCreateShader(type);
   27.85 +	}
   27.86 +
   27.87 +	info_log("compiling shader: %s... ", name.c_str());
   27.88 +
   27.89 +	glShaderSource(sdr, sizeof src_arr / sizeof *src_arr, src_arr, 0);
   27.90 +	glCompileShader(sdr);
   27.91 +
   27.92 +	int status;
   27.93 +	glGetShaderiv(sdr, GL_COMPILE_STATUS, &status);
   27.94 +
   27.95 +	info_log(status ? "success\n" : "failed\n");
   27.96 +
   27.97 +	int info_len;
   27.98 +	glGetShaderiv(sdr, GL_INFO_LOG_LENGTH, &info_len);
   27.99 +	if(info_len > 1) {
  27.100 +		char *buf = (char*)alloca(info_len);
  27.101 +		glGetShaderInfoLog(sdr, info_len, 0, buf);
  27.102 +		buf[info_len - 1] = 0;
  27.103 +
  27.104 +		if(status) {
  27.105 +			info_log("%s\n", buf);
  27.106 +		} else {
  27.107 +			error_log("%s\n", buf);
  27.108 +		}
  27.109 +	}
  27.110 +
  27.111 +	return status == GL_TRUE;
  27.112 +}
  27.113 +
  27.114 +void Shader::destroy()
  27.115 +{
  27.116 +	if(sdr) {
  27.117 +		glDeleteShader(sdr);
  27.118 +	}
  27.119 +	sdr = type = 0;
  27.120 +
  27.121 +	src.clear();
  27.122 +	name.clear();
  27.123 +}
  27.124 +
  27.125 +static std::string read_source(const char *fname)
  27.126 +{
  27.127 +	FILE *fp;
  27.128 +
  27.129 +	if(!(fp = fopen(fname, "rb"))) {
  27.130 +		error_log("failed to load shader: %s: %s\n", fname, strerror(errno));
  27.131 +		return std::string();
  27.132 +	}
  27.133 +
  27.134 +	fseek(fp, 0, SEEK_END);
  27.135 +	long sz = ftell(fp);
  27.136 +	rewind(fp);
  27.137 +
  27.138 +	char *src = (char*)alloca(sz + 1);
  27.139 +	if(fread(src, 1, sz, fp) < (size_t)sz) {
  27.140 +		error_log("failed to load shader: %s: %s\n", fname, strerror(errno));
  27.141 +		fclose(fp);
  27.142 +		delete [] src;
  27.143 +		return std::string();
  27.144 +	}
  27.145 +	src[sz] = 0;
  27.146 +	fclose(fp);
  27.147 +
  27.148 +	return std::string(src);
  27.149 +}
  27.150 +
  27.151 +bool Shader::load(const char *fname, unsigned int type)
  27.152 +{
  27.153 +	std::string src = read_source(fname);
  27.154 +	if(src.empty()) {
  27.155 +		return false;
  27.156 +	}
  27.157 +	set_name(fname);
  27.158 +	return create(src.c_str(), type);
  27.159 +}
  27.160 +
  27.161 +// ---- shader program ----
  27.162 +ShaderProg::ShaderProg()
  27.163 +{
  27.164 +	prog = 0;
  27.165 +	must_link = true;
  27.166 +}
  27.167 +
  27.168 +ShaderProg::~ShaderProg()
  27.169 +{
  27.170 +	destroy();
  27.171 +}
  27.172 +
  27.173 +unsigned int ShaderProg::get_id() const
  27.174 +{
  27.175 +	return prog;
  27.176 +}
  27.177 +
  27.178 +bool ShaderProg::create(const char *src, unsigned int type, ...)
  27.179 +{
  27.180 +	va_list ap;
  27.181 +
  27.182 +	va_start(ap, type);
  27.183 +	bool res = create(src, type, ap);
  27.184 +	va_end(ap);
  27.185 +
  27.186 +	return res;
  27.187 +}
  27.188 +
  27.189 +bool ShaderProg::create(const char *src, unsigned int type, va_list ap)
  27.190 +{
  27.191 +	destroy();
  27.192 +	prog = glCreateProgram();
  27.193 +
  27.194 +	while(src) {
  27.195 +		Shader *sdr = new Shader;
  27.196 +		if(!sdr->create(src, type)) {
  27.197 +			va_end(ap);
  27.198 +			return false;
  27.199 +		}
  27.200 +		add_shader(sdr);
  27.201 +		src = va_arg(ap, const char*);
  27.202 +		type = va_arg(ap, unsigned int);
  27.203 +	}
  27.204 +
  27.205 +	return link();
  27.206 +}
  27.207 +
  27.208 +bool ShaderProg::create(const char *vsrc, const char *psrc)
  27.209 +{
  27.210 +	return create(VSDR(vsrc), PSDR(psrc), 0);
  27.211 +}
  27.212 +
  27.213 +bool ShaderProg::create(Shader *sdr, ...)
  27.214 +{
  27.215 +	va_list ap;
  27.216 +
  27.217 +	va_start(ap, sdr);
  27.218 +	bool res = create(sdr, ap);
  27.219 +	va_end(ap);
  27.220 +
  27.221 +	return res;
  27.222 +}
  27.223 +
  27.224 +bool ShaderProg::create(Shader *sdr, va_list ap)
  27.225 +{
  27.226 +	destroy();
  27.227 +	prog = glCreateProgram();
  27.228 +
  27.229 +	while(sdr) {
  27.230 +		add_shader(sdr);
  27.231 +		sdr = va_arg(ap, Shader*);
  27.232 +	}
  27.233 +	return link();
  27.234 +}
  27.235 +
  27.236 +bool ShaderProg::create(Shader *vsdr, Shader *psdr)
  27.237 +{
  27.238 +	return create(vsdr, psdr, 0);
  27.239 +}
  27.240 +
  27.241 +void ShaderProg::destroy()
  27.242 +{
  27.243 +	if(prog) {
  27.244 +		glDeleteProgram(prog);
  27.245 +	}
  27.246 +	prog = 0;
  27.247 +
  27.248 +	shaders.clear();
  27.249 +	// don't actually destroy the shaders, let the ShaderSet own them
  27.250 +}
  27.251 +
  27.252 +bool ShaderProg::load(const char *fname, unsigned int type, ...)
  27.253 +{
  27.254 +	va_list ap;
  27.255 +	va_start(ap, type);
  27.256 +	bool res = load(fname, type, ap);
  27.257 +	va_end(ap);
  27.258 +
  27.259 +	return res;
  27.260 +}
  27.261 +
  27.262 +bool ShaderProg::load(const char *fname, unsigned int type, va_list ap)
  27.263 +{
  27.264 +	destroy();
  27.265 +	prog = glCreateProgram();
  27.266 +
  27.267 +	while(fname) {
  27.268 +		Shader *sdr = new Shader;
  27.269 +		if(!sdr->load(fname, type)) {
  27.270 +			delete sdr;
  27.271 +			return false;
  27.272 +		}
  27.273 +		add_shader(sdr);
  27.274 +
  27.275 +		if((fname = va_arg(ap, const char*))) {
  27.276 +			type = va_arg(ap, unsigned int);
  27.277 +		}
  27.278 +	}
  27.279 +
  27.280 +	return link();
  27.281 +}
  27.282 +
  27.283 +bool ShaderProg::load(const char *vfname, const char *pfname)
  27.284 +{
  27.285 +	return load(VSDR(vfname), PSDR(pfname), 0);
  27.286 +}
  27.287 +
  27.288 +void ShaderProg::add_shader(Shader *sdr)
  27.289 +{
  27.290 +	glAttachShader(prog, sdr->get_id());
  27.291 +}
  27.292 +
  27.293 +bool ShaderProg::link() const
  27.294 +{
  27.295 +	bind_standard_attr(this);
  27.296 +
  27.297 +	CHECKGLERR;
  27.298 +	info_log("linking program ... ");
  27.299 +	glLinkProgram(prog);
  27.300 +
  27.301 +	int status;
  27.302 +	glGetProgramiv(prog, GL_LINK_STATUS, &status);
  27.303 +
  27.304 +	info_log(status ? "success\n" : "failed\n");
  27.305 +
  27.306 +	int info_len;
  27.307 +	glGetProgramiv(prog, GL_INFO_LOG_LENGTH, &info_len);
  27.308 +	if(info_len > 1) {
  27.309 +		char *buf = (char*)alloca(info_len);
  27.310 +		glGetProgramInfoLog(prog, info_len, 0, buf);
  27.311 +		buf[info_len - 1] = 0;
  27.312 +
  27.313 +		if(status) {
  27.314 +			info_log("%s\n", buf);
  27.315 +		} else {
  27.316 +			error_log("%s\n", buf);
  27.317 +		}
  27.318 +	}
  27.319 +
  27.320 +	if(status) {
  27.321 +		must_link = false;
  27.322 +		cache_state_uniforms();
  27.323 +		return true;
  27.324 +	}
  27.325 +	return false;
  27.326 +}
  27.327 +
  27.328 +void ShaderProg::bind() const
  27.329 +{
  27.330 +	CHECKGLERR;
  27.331 +	if(must_link) {
  27.332 +		if(!link()) {
  27.333 +			return;
  27.334 +		}
  27.335 +	}
  27.336 +	CHECKGLERR;
  27.337 +	glUseProgram(prog);
  27.338 +	ShaderProg::current = (ShaderProg*)this;
  27.339 +
  27.340 +	setup_state_uniforms();
  27.341 +}
  27.342 +
  27.343 +
  27.344 +int ShaderProg::get_attrib_location(const char *name) const
  27.345 +{
  27.346 +	glUseProgram(prog);
  27.347 +	return glGetAttribLocation(prog, name);
  27.348 +}
  27.349 +
  27.350 +void ShaderProg::set_attrib_location(const char *name, int loc) const
  27.351 +{
  27.352 +	glBindAttribLocation(prog, loc, name);
  27.353 +	must_link = true;
  27.354 +}
  27.355 +
  27.356 +int ShaderProg::get_uniform_location(const char *name) const
  27.357 +{
  27.358 +	glUseProgram(prog);
  27.359 +	return glGetUniformLocation(prog, name);
  27.360 +}
  27.361 +
  27.362 +bool ShaderProg::set_uniform(int loc, int val) const
  27.363 +{
  27.364 +	glUseProgram(prog);
  27.365 +	if(loc >= 0) {
  27.366 +		glUniform1i(loc, val);
  27.367 +		return true;
  27.368 +	}
  27.369 +	return false;
  27.370 +}
  27.371 +
  27.372 +bool ShaderProg::set_uniform(int loc, float val) const
  27.373 +{
  27.374 +	glUseProgram(prog);
  27.375 +	if(loc >= 0) {
  27.376 +		glUniform1f(loc, val);
  27.377 +		return true;
  27.378 +	}
  27.379 +	return false;
  27.380 +}
  27.381 +
  27.382 +bool ShaderProg::set_uniform(int loc, const Vector2 &v) const
  27.383 +{
  27.384 +	glUseProgram(prog);
  27.385 +	if(loc >= 0) {
  27.386 +		glUniform2f(loc, v.x, v.y);
  27.387 +		return true;
  27.388 +	}
  27.389 +	return false;
  27.390 +}
  27.391 +
  27.392 +bool ShaderProg::set_uniform(int loc, const Vector3 &v) const
  27.393 +{
  27.394 +	glUseProgram(prog);
  27.395 +	if(loc >= 0) {
  27.396 +		glUniform3f(loc, v.x, v.y, v.z);
  27.397 +		return true;
  27.398 +	}
  27.399 +	return false;
  27.400 +}
  27.401 +
  27.402 +bool ShaderProg::set_uniform(int loc, const Vector4 &v) const
  27.403 +{
  27.404 +	glUseProgram(prog);
  27.405 +	if(loc >= 0) {
  27.406 +		glUniform4f(loc, v.x, v.y, v.z, v.w);
  27.407 +		return true;
  27.408 +	}
  27.409 +	return false;
  27.410 +}
  27.411 +
  27.412 +bool ShaderProg::set_uniform(int loc, const Matrix3x3 &m) const
  27.413 +{
  27.414 +	glUseProgram(prog);
  27.415 +	if(loc >= 0) {
  27.416 +		glUniformMatrix3fv(loc, 1, GL_TRUE, m[0]);
  27.417 +		return true;
  27.418 +	}
  27.419 +	return false;
  27.420 +}
  27.421 +
  27.422 +bool ShaderProg::set_uniform(int loc, const Matrix4x4 &m) const
  27.423 +{
  27.424 +	glUseProgram(prog);
  27.425 +	if(loc >= 0) {
  27.426 +		glUniformMatrix4fv(loc, 1, GL_TRUE, m[0]);
  27.427 +		return true;
  27.428 +	}
  27.429 +	return false;
  27.430 +}
  27.431 +
  27.432 +
  27.433 +bool ShaderProg::set_uniform(const char *name, int val) const
  27.434 +{
  27.435 +	return set_uniform(get_uniform_location(name), val);
  27.436 +}
  27.437 +
  27.438 +bool ShaderProg::set_uniform(const char *name, float val) const
  27.439 +{
  27.440 +	return set_uniform(get_uniform_location(name), val);
  27.441 +}
  27.442 +
  27.443 +bool ShaderProg::set_uniform(const char *name, const Vector2 &v) const
  27.444 +{
  27.445 +	return set_uniform(get_uniform_location(name), v);
  27.446 +}
  27.447 +
  27.448 +bool ShaderProg::set_uniform(const char *name, const Vector3 &v) const
  27.449 +{
  27.450 +	return set_uniform(get_uniform_location(name), v);
  27.451 +}
  27.452 +
  27.453 +bool ShaderProg::set_uniform(const char *name, const Vector4 &v) const
  27.454 +{
  27.455 +	return set_uniform(get_uniform_location(name), v);
  27.456 +}
  27.457 +
  27.458 +bool ShaderProg::set_uniform(const char *name, const Matrix3x3 &m) const
  27.459 +{
  27.460 +	return set_uniform(get_uniform_location(name), m);
  27.461 +}
  27.462 +
  27.463 +bool ShaderProg::set_uniform(const char *name, const Matrix4x4 &m) const
  27.464 +{
  27.465 +	return set_uniform(get_uniform_location(name), m);
  27.466 +}
  27.467 +
  27.468 +static StType unist_type(GLenum type)
  27.469 +{
  27.470 +	switch(type) {
  27.471 +	case GL_FLOAT:
  27.472 +		return ST_FLOAT;
  27.473 +	case GL_FLOAT_VEC2:
  27.474 +		return ST_FLOAT2;
  27.475 +	case GL_FLOAT_VEC3:
  27.476 +		return ST_FLOAT3;
  27.477 +	case GL_FLOAT_VEC4:
  27.478 +		return ST_FLOAT4;
  27.479 +	case GL_INT:
  27.480 +	case GL_SAMPLER_2D:
  27.481 +	case GL_SAMPLER_CUBE:
  27.482 +#if !GL_ES_VERSION_2_0
  27.483 +	case GL_SAMPLER_1D:
  27.484 +	case GL_SAMPLER_3D:
  27.485 +	case GL_SAMPLER_1D_SHADOW:
  27.486 +	case GL_SAMPLER_2D_SHADOW:
  27.487 +#endif
  27.488 +		return ST_INT;
  27.489 +	case GL_INT_VEC2:
  27.490 +		return ST_INT2;
  27.491 +	case GL_INT_VEC3:
  27.492 +		return ST_INT3;
  27.493 +	case GL_INT_VEC4:
  27.494 +		return ST_INT4;
  27.495 +	case GL_FLOAT_MAT3:
  27.496 +		return ST_MATRIX3;
  27.497 +	case GL_FLOAT_MAT4:
  27.498 +		return ST_MATRIX4;
  27.499 +	default:
  27.500 +		break;
  27.501 +	}
  27.502 +	return ST_UNKNOWN;
  27.503 +}
  27.504 +
  27.505 +void ShaderProg::cache_state_uniforms() const
  27.506 +{
  27.507 +	if(!glIsProgram(prog)) {
  27.508 +		return;
  27.509 +	}
  27.510 +
  27.511 +	int num_uni;
  27.512 +	glGetProgramiv(prog, GL_ACTIVE_UNIFORMS, &num_uni);
  27.513 +
  27.514 +	char name[256];
  27.515 +	for(int i=0; i<num_uni; i++) {
  27.516 +		GLint sz;
  27.517 +		GLenum type;
  27.518 +		glGetActiveUniform(prog, i, sizeof name - 1, 0, &sz, &type, name);
  27.519 +
  27.520 +		if(strstr(name, "st_") == name) {
  27.521 +			StateLocCache s;
  27.522 +			s.sidx = add_unistate(name, unist_type(type));
  27.523 +			s.loc = glGetUniformLocation(prog, name);
  27.524 +			stloc_cache.push_back(s);
  27.525 +		}
  27.526 +	}
  27.527 +}
  27.528 +
  27.529 +void ShaderProg::setup_state_uniforms() const
  27.530 +{
  27.531 +	for(size_t i=0; i<stloc_cache.size(); i++) {
  27.532 +		setup_unistate(stloc_cache[i].sidx, this, stloc_cache[i].loc);
  27.533 +		CHECKGLERR;
  27.534 +	}
  27.535 +}
  27.536 +
  27.537 +
  27.538 +// ---- ShaderSet ----
  27.539 +static Shader *load_shader(const char *fname, unsigned int type)
  27.540 +{
  27.541 +	Shader *sdr = new Shader;
  27.542 +	if(!sdr->load(fname, type)) {
  27.543 +		delete sdr;
  27.544 +		return 0;
  27.545 +	}
  27.546 +	return sdr;
  27.547 +}
  27.548 +
  27.549 +static Shader *load_vertex_shader(const char *fname)
  27.550 +{
  27.551 +	return load_shader(fname, GL_VERTEX_SHADER);
  27.552 +}
  27.553 +
  27.554 +static Shader *load_pixel_shader(const char *fname)
  27.555 +{
  27.556 +	return load_shader(fname, GL_FRAGMENT_SHADER);
  27.557 +}
  27.558 +
  27.559 +#ifdef HAVE_GEOMETRY_SHADER
  27.560 +static Shader *load_geom_shader(const char *fname)
  27.561 +{
  27.562 +	return load_shader(fname, GL_GEOMETRY_SHADER);
  27.563 +}
  27.564 +#endif
  27.565 +
  27.566 +#ifdef HAVE_TESSELATION_SHADER
  27.567 +static Shader *load_tc_shader(const char *fname)
  27.568 +{
  27.569 +	return load_shader(fname, GL_TESS_CONTROL_SHADER);
  27.570 +}
  27.571 +
  27.572 +static Shader *load_te_shader(const char *fname)
  27.573 +{
  27.574 +	return load_shader(fname, GL_TESS_EVALUATION_SHADER);
  27.575 +}
  27.576 +#endif
  27.577 +
  27.578 +static void destroy_shader(Shader *sdr)
  27.579 +{
  27.580 +	delete sdr;
  27.581 +}
  27.582 +
  27.583 +ShaderSet::ShaderSet(unsigned int type)
  27.584 +	: DataSet<Shader*>(0, destroy_shader)
  27.585 +{
  27.586 +	this->type = type;
  27.587 +
  27.588 +	switch(type) {
  27.589 +	case GL_VERTEX_SHADER:
  27.590 +		load = load_vertex_shader;
  27.591 +		break;
  27.592 +
  27.593 +	case GL_FRAGMENT_SHADER:
  27.594 +		load = load_pixel_shader;
  27.595 +		break;
  27.596 +
  27.597 +#ifdef HAVE_GEOMETRY_SHADER
  27.598 +	case GL_GEOMETRY_SHADER:
  27.599 +		load = load_geom_shader;
  27.600 +		break;
  27.601 +#endif
  27.602 +
  27.603 +#ifdef HAVE_TESSELATION_SHADER
  27.604 +	case GL_TESS_CONTROL_SHADER:
  27.605 +		load = load_tc_shader;
  27.606 +		break;
  27.607 +
  27.608 +	case GL_TESS_EVALUATION_SHADER:
  27.609 +		load = load_te_shader;
  27.610 +		break;
  27.611 +#endif
  27.612 +
  27.613 +	default:
  27.614 +		error_log("ShaderSet constructed with invalid shader type!\n");
  27.615 +	}
  27.616 +}
  27.617 +
  27.618 +static struct { const char *name; int loc; } attr_loc[] = {
  27.619 +	{"attr_vertex", MESH_ATTR_VERTEX},
  27.620 +	{"attr_normal", MESH_ATTR_NORMAL},
  27.621 +	{"attr_tangent", MESH_ATTR_TANGENT},
  27.622 +	{"attr_texcoord", MESH_ATTR_TEXCOORD},
  27.623 +	{"attr_color", MESH_ATTR_COLOR},
  27.624 +	{"attr_boneweights", MESH_ATTR_BONEWEIGHTS},
  27.625 +	{"attr_boneidx", MESH_ATTR_BONEIDX}
  27.626 +};
  27.627 +
  27.628 +static void bind_standard_attr(const ShaderProg *prog)
  27.629 +{
  27.630 +	// we must link once to find out which are the active attributes
  27.631 +	glLinkProgram(prog->get_id());
  27.632 +
  27.633 +	int num_attr;
  27.634 +	glGetProgramiv(prog->get_id(), GL_ACTIVE_ATTRIBUTES, &num_attr);
  27.635 +
  27.636 +	char name[256];
  27.637 +	for(int i=0; i<num_attr; i++) {
  27.638 +		GLint sz;
  27.639 +		GLenum type;
  27.640 +		glGetActiveAttrib(prog->get_id(), i, sizeof name - 1, 0, &sz, &type, name);
  27.641 +
  27.642 +		for(int j=0; j<(int)(sizeof attr_loc / sizeof *attr_loc); j++) {
  27.643 +			if(strcmp(name, attr_loc[j].name) == 0) {
  27.644 +				prog->set_attrib_location(name, attr_loc[j].loc);
  27.645 +			}
  27.646 +		}
  27.647 +	}
  27.648 +}
  27.649 +
  27.650 +
  27.651 +/*
  27.652 +static const char *strtype(unsigned int type)
  27.653 +{
  27.654 +	switch(type) {
  27.655 +	case GL_VERTEX_SHADER:
  27.656 +		return "vertex";
  27.657 +	case GL_FRAGMENT_SHADER:
  27.658 +		return "fragment";
  27.659 +#ifdef HAVE_GEOMETRY_SHADER
  27.660 +	case GL_GEOMETRY_SHADER:
  27.661 +		return "geometry";
  27.662 +#endif
  27.663 +#ifdef HAVE_TESSELATION_SHADER
  27.664 +	case GL_TESS_CONTROL_SHADER:
  27.665 +		return "tesselation control";
  27.666 +	case GL_TESS_EVALUATION_SHADER:
  27.667 +		return "tesselation evaluation";
  27.668 +#endif
  27.669 +	default:
  27.670 +		break;
  27.671 +	}
  27.672 +	return "<unknown>";
  27.673 +}
  27.674 +*/
    28.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    28.2 +++ b/src/shader.h	Sun Aug 24 09:41:24 2014 +0300
    28.3 @@ -0,0 +1,150 @@
    28.4 +#ifndef SHADER_H_
    28.5 +#define SHADER_H_
    28.6 +
    28.7 +#include <vector>
    28.8 +#include <string>
    28.9 +#include "vmath/vmath.h"
   28.10 +#include "opengl.h"
   28.11 +#include "dataset.h"
   28.12 +
   28.13 +class ShaderProg;
   28.14 +
   28.15 +
   28.16 +void bind_shader(const ShaderProg *sdr);
   28.17 +const ShaderProg *get_current_shader();
   28.18 +
   28.19 +
   28.20 +class Shader {
   28.21 +private:
   28.22 +	unsigned int sdr;
   28.23 +	unsigned int type;
   28.24 +	std::string name, src;
   28.25 +
   28.26 +public:
   28.27 +	Shader();
   28.28 +	~Shader();
   28.29 +
   28.30 +	unsigned int get_id() const;
   28.31 +
   28.32 +	void set_name(const char *name);
   28.33 +	const char *get_name() const;
   28.34 +
   28.35 +	bool create(const char *src, unsigned int type);
   28.36 +	void destroy();
   28.37 +
   28.38 +	bool load(const char *fname, unsigned int type);
   28.39 +
   28.40 +
   28.41 +	// these functions are only meant to be used by the ShaderSet
   28.42 +	static Shader *create_shader();
   28.43 +	static bool load_shader(Shader *sdr, const char *fname);
   28.44 +	static bool done_shader(Shader *sdr, unsigned int type);
   28.45 +	static void destroy_shader(Shader *sdr);
   28.46 +};
   28.47 +
   28.48 +#define VSDR(s)		s, GL_VERTEX_SHADER
   28.49 +#define FSDR(s)		s, GL_FRAGMENT_SHADER
   28.50 +#define PSDR(s)		FSDR(s)
   28.51 +#define GSDR(s)		s, GL_GEOMETRY_SHADER
   28.52 +#define TCSDR(s)	s, GL_TESS_CONTROL_SHADER
   28.53 +#define TESDR(s)	s, GL_TESS_EVALUATION_SHADER
   28.54 +
   28.55 +class ShaderProg {
   28.56 +private:
   28.57 +	unsigned int prog;
   28.58 +	mutable bool must_link;
   28.59 +	std::vector<Shader*> shaders;
   28.60 +
   28.61 +	struct StateLocCache { int sidx, loc; };
   28.62 +	/** a cache of all st_ prefixed uniform locations and their corresponding
   28.63 +	 * index in the global uniform state vector (see unistate.h)
   28.64 +	 */
   28.65 +	mutable std::vector<StateLocCache> stloc_cache;
   28.66 +
   28.67 +	void cache_state_uniforms() const;
   28.68 +	void setup_state_uniforms() const;
   28.69 +
   28.70 +public:
   28.71 +	static ShaderProg *current;
   28.72 +
   28.73 +	ShaderProg();
   28.74 +	~ShaderProg();
   28.75 +
   28.76 +	/// returns the OpenGL object id for this shader program
   28.77 +	unsigned int get_id() const;
   28.78 +
   28.79 +	/** takes a series of shaders, and constructs a program object by linking
   28.80 +	 * them together. Terminate with a null pointer (don't use 0!) */
   28.81 +	bool create(Shader *sdr, ...);
   28.82 +	/// same as above, but with a va_list instead of variable arguments.
   28.83 +	bool create(Shader *sdr, va_list ap);
   28.84 +	/** takes two shaders (vertex and pixel) and constructs a program object by
   28.85 +	 * linking them together. Either one can be null. */
   28.86 +	bool create(Shader *vsdr, Shader *psdr);
   28.87 +
   28.88 +	/** takes a series of shader source/shader type pairs and constructs a program
   28.89 +	 * object by linking them together. Terminate with a null pointer (don't use 0!)
   28.90 +	 * You can use the VSDR, PSDR, GSDR, TCSDR, TESDR convenience macros for passing
   28.91 +	 * the pairs.
   28.92 +	 * Example: create(VSDR(vsrc0), VSDR(vsrc1), PSDR(psrc), NULL);
   28.93 +	 */
   28.94 +	bool create(const char *src, unsigned int type, ...);
   28.95 +	/// same as above, but with a va_list instead of variable arguments.
   28.96 +	bool create(const char *src, unsigned int type, va_list ap);
   28.97 +	/** takes two shaders source strings (vertex and pixel) and constructs
   28.98 +	 * a program object by linking them together. Either one can be null. */
   28.99 +	bool create(const char *vsrc, const char *psrc);
  28.100 +
  28.101 +	void destroy();
  28.102 +
  28.103 +	/** takes a series of shader filename/shader type pairs, loads the shaders and
  28.104 +	 * constructs a program object by linking them together. Terminate with a null
  28.105 +	 * pointer (don't use 0!). You can use the VSDR, PSDR, GSDR, TCSDR, TESDR convenience
  28.106 +	 * macros for passing the pairs.
  28.107 +	 * Example: load(VSDR("vsdr1.glsl"), VSDR("vsdr2.glsl"), PSDR("pixel.glsl"), NULL);
  28.108 +	 */
  28.109 +	bool load(const char *fname, unsigned int type, ...);
  28.110 +	/// same as above, but with a va_list instead of variable arguments.
  28.111 +	bool load(const char *fname, unsigned int type, va_list ap);
  28.112 +	/** takes the filenames of two shader files (vertex and pixel), loads them and
  28.113 +	 * constructs a program object by linking them together. Either one can be null */
  28.114 +	bool load(const char *vsrc, const char *psrc);
  28.115 +
  28.116 +	void add_shader(Shader *sdr);
  28.117 +	bool link() const;
  28.118 +
  28.119 +	void bind() const;
  28.120 +
  28.121 +	int get_attrib_location(const char *name) const;
  28.122 +	void set_attrib_location(const char *name, int loc) const;
  28.123 +
  28.124 +	int get_uniform_location(const char *name) const;
  28.125 +
  28.126 +	bool set_uniform(int loc, int val) const;
  28.127 +	bool set_uniform(int loc, float val) const;
  28.128 +	bool set_uniform(int loc, const Vector2 &v) const;
  28.129 +	bool set_uniform(int loc, const Vector3 &v) const;
  28.130 +	bool set_uniform(int loc, const Vector4 &v) const;
  28.131 +	bool set_uniform(int loc, const Matrix3x3 &m) const;
  28.132 +	bool set_uniform(int loc, const Matrix4x4 &m) const;
  28.133 +
  28.134 +	bool set_uniform(const char *name, int val) const;
  28.135 +	bool set_uniform(const char *name, float val) const;
  28.136 +	bool set_uniform(const char *name, const Vector2 &v) const;
  28.137 +	bool set_uniform(const char *name, const Vector3 &v) const;
  28.138 +	bool set_uniform(const char *name, const Vector4 &v) const;
  28.139 +	bool set_uniform(const char *name, const Matrix3x3 &m) const;
  28.140 +	bool set_uniform(const char *name, const Matrix4x4 &m) const;
  28.141 +
  28.142 +	friend void setup_unistate(const ShaderProg*);
  28.143 +};
  28.144 +
  28.145 +class ShaderSet : public DataSet<Shader*> {
  28.146 +private:
  28.147 +	unsigned int type;
  28.148 +
  28.149 +public:
  28.150 +	ShaderSet(unsigned int type);
  28.151 +};
  28.152 +
  28.153 +#endif	// SHADER_H_
    29.1 --- a/src/texture.cc	Sat Aug 23 12:03:29 2014 +0300
    29.2 +++ b/src/texture.cc	Sun Aug 24 09:41:24 2014 +0300
    29.3 @@ -64,18 +64,16 @@
    29.4  	return img;
    29.5  }
    29.6  
    29.7 +unsigned int Texture::get_type() const
    29.8 +{
    29.9 +	return type;
   29.10 +}
   29.11 +
   29.12  unsigned int Texture::get_texture_id() const
   29.13  {
   29.14  	return tex;
   29.15  }
   29.16  
   29.17 -void Texture::bind(int tunit) const
   29.18 -{
   29.19 -	glActiveTextureARB(GL_TEXTURE0_ARB + tunit);
   29.20 -	glBindTexture(type, tex);
   29.21 -	glActiveTextureARB(GL_TEXTURE0_ARB);
   29.22 -}
   29.23 -
   29.24  bool Texture::load(const char *fname)
   29.25  {
   29.26  	Image image;
   29.27 @@ -86,6 +84,26 @@
   29.28  	return true;
   29.29  }
   29.30  
   29.31 +void bind_texture(const Texture *tex, int tunit)
   29.32 +{
   29.33 +	static unsigned int cur_tex_type[8] = {
   29.34 +		GL_TEXTURE_2D, GL_TEXTURE_2D, GL_TEXTURE_2D, GL_TEXTURE_2D,
   29.35 +		GL_TEXTURE_2D, GL_TEXTURE_2D, GL_TEXTURE_2D, GL_TEXTURE_2D
   29.36 +	};
   29.37 +
   29.38 +	glActiveTextureARB(GL_TEXTURE0_ARB + tunit);
   29.39 +	if(tex) {
   29.40 +		glBindTexture(tex->get_type(), tex->get_texture_id());
   29.41 +		glEnable(tex->get_type());
   29.42 +	} else {
   29.43 +		glDisable(cur_tex_type[tunit]);
   29.44 +	}
   29.45 +	glActiveTextureARB(GL_TEXTURE0_ARB);
   29.46 +
   29.47 +	cur_tex_type[tunit] = tex->get_type();
   29.48 +}
   29.49 +
   29.50 +
   29.51  unsigned int next_pow2(unsigned int x)
   29.52  {
   29.53  	x -= 1;
    30.1 --- a/src/texture.h	Sat Aug 23 12:03:29 2014 +0300
    30.2 +++ b/src/texture.h	Sun Aug 24 09:41:24 2014 +0300
    30.3 @@ -23,12 +23,14 @@
    30.4  	Image &get_image();
    30.5  	const Image &get_image() const;
    30.6  
    30.7 +	unsigned int get_type() const;
    30.8  	unsigned int get_texture_id() const;
    30.9 -	void bind(int tunit = 0) const;
   30.10  
   30.11  	bool load(const char *fname);
   30.12  };
   30.13  
   30.14 +void bind_texture(const Texture *tex, int unit = 0);
   30.15 +
   30.16  unsigned int next_pow2(unsigned int x);
   30.17  
   30.18  #endif	// TEXTURE_H_
    31.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    31.2 +++ b/src/unistate.cc	Sun Aug 24 09:41:24 2014 +0300
    31.3 @@ -0,0 +1,678 @@
    31.4 +#include <map>
    31.5 +#include <vector>
    31.6 +#include "unistate.h"
    31.7 +#include "shader.h"
    31.8 +#include "logger.h"
    31.9 +
   31.10 +struct StateItem {
   31.11 +	StType type;
   31.12 +
   31.13 +	union {
   31.14 +		int ival[4];
   31.15 +		float fval[16];
   31.16 +	};
   31.17 +	int transpose;	// for matrices
   31.18 +};
   31.19 +
   31.20 +static const char *typestr(StType type);
   31.21 +static int type_nelem(StType type);
   31.22 +static StType float_type(int elem);
   31.23 +static StType int_type(int elem);
   31.24 +
   31.25 +std::vector<StateItem> state;
   31.26 +std::map<std::string, int> stateidx;
   31.27 +
   31.28 +
   31.29 +int add_unistate(const char *name, StType type)
   31.30 +{
   31.31 +	static const float ident3[] = {1, 0, 0, 0, 1, 0, 0, 0, 1};
   31.32 +	static const float ident4[] = {1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1};
   31.33 +
   31.34 +	if(stateidx.find(name) != stateidx.end()) {
   31.35 +		return stateidx[name];
   31.36 +	}
   31.37 +
   31.38 +	StateItem sitem;
   31.39 +	memset(&sitem, 0, sizeof sitem);
   31.40 +	sitem.type = type;
   31.41 +
   31.42 +	// initialize to a reasonable default value
   31.43 +	switch(type) {
   31.44 +	case ST_MATRIX3:
   31.45 +		memcpy(sitem.fval, ident3, sizeof ident3);
   31.46 +		break;
   31.47 +
   31.48 +	case ST_MATRIX4:
   31.49 +		memcpy(sitem.fval, ident4, sizeof ident4);
   31.50 +		break;
   31.51 +
   31.52 +	default:
   31.53 +		break;	// in all other cases leave it zero (see memset above)
   31.54 +	}
   31.55 +
   31.56 +	int sidx = state.size();
   31.57 +	state.push_back(sitem);
   31.58 +	stateidx[name] = sidx;
   31.59 +
   31.60 +	debug_log("adding uniform state [%d]: %s %s\n", sidx, typestr(sitem.type), name);
   31.61 +
   31.62 +	return sidx;
   31.63 +}
   31.64 +
   31.65 +int get_unistate_index(const char *name)
   31.66 +{
   31.67 +	std::map<std::string, int>::const_iterator it = stateidx.find(name);
   31.68 +	if(it != stateidx.end()) {
   31.69 +		return it->second;
   31.70 +	}
   31.71 +	return -1;
   31.72 +}
   31.73 +
   31.74 +#define CHECK_INDEX(i)	\
   31.75 +	if(i < 0 || i >= (int)state.size()) return
   31.76 +
   31.77 +#define CHECK_COUNT(count, type) \
   31.78 +	do { \
   31.79 +		int max_elem = type_nelem(type); \
   31.80 +		if(!(count) || (count) > max_elem) { \
   31.81 +			count = max_elem; \
   31.82 +		} \
   31.83 +	} while(0)
   31.84 +
   31.85 +void set_unistate(int sidx, const int *val, int count)
   31.86 +{
   31.87 +	CHECK_INDEX(sidx);
   31.88 +	CHECK_COUNT(count, state[sidx].type);
   31.89 +
   31.90 +	memcpy(state[sidx].ival, val, count * sizeof *state[sidx].ival);
   31.91 +}
   31.92 +
   31.93 +void set_unistate(int sidx, const float *val, int count)
   31.94 +{
   31.95 +	CHECK_INDEX(sidx);
   31.96 +	CHECK_COUNT(count, state[sidx].type);
   31.97 +
   31.98 +	memcpy(state[sidx].fval, val, count * sizeof *state[sidx].fval);
   31.99 +	state[sidx].transpose = 0;
  31.100 +}
  31.101 +
  31.102 +void get_unistate(int sidx, int *val, int count)
  31.103 +{
  31.104 +	CHECK_INDEX(sidx);
  31.105 +	CHECK_COUNT(count, state[sidx].type);
  31.106 +
  31.107 +	memcpy(val, state[sidx].ival, count * sizeof *val);
  31.108 +}
  31.109 +
  31.110 +void get_unistate(int sidx, float *val, int count)
  31.111 +{
  31.112 +	CHECK_INDEX(sidx);
  31.113 +	CHECK_COUNT(count, state[sidx].type);
  31.114 +
  31.115 +	memcpy(val, state[sidx].fval, count * sizeof *val);
  31.116 +}
  31.117 +
  31.118 +void set_unistate(int sidx, int val)
  31.119 +{
  31.120 +	set_unistate(sidx, &val, 1);
  31.121 +}
  31.122 +
  31.123 +void set_unistate(int sidx, float val)
  31.124 +{
  31.125 +	set_unistate(sidx, &val, 1);
  31.126 +}
  31.127 +
  31.128 +void set_unistate(int sidx, const Vector2 &vec)
  31.129 +{
  31.130 +	set_unistate(sidx, &vec.x, 2);
  31.131 +}
  31.132 +
  31.133 +void set_unistate(int sidx, const Vector3 &vec)
  31.134 +{
  31.135 +	set_unistate(sidx, &vec.x, 3);
  31.136 +}
  31.137 +
  31.138 +void set_unistate(int sidx, const Vector4 &vec)
  31.139 +{
  31.140 +	set_unistate(sidx, &vec.x, 4);
  31.141 +}
  31.142 +
  31.143 +void set_unistate(int sidx, const Matrix3x3 &mat)
  31.144 +{
  31.145 +	set_unistate(sidx, mat[0], 9);
  31.146 +	state[sidx].transpose = 1;
  31.147 +}
  31.148 +
  31.149 +void set_unistate(int sidx, const Matrix4x4 &mat)
  31.150 +{
  31.151 +	set_unistate(sidx, mat[0], 16);
  31.152 +	state[sidx].transpose = 1;
  31.153 +}
  31.154 +
  31.155 +
  31.156 +int set_unistate(const char *name, int *val, int count)
  31.157 +{
  31.158 +	int sidx = get_unistate_index(name);
  31.159 +	if(sidx < 0) {
  31.160 +		StType type = int_type(count);
  31.161 +		if(type == ST_UNKNOWN) {
  31.162 +			error_log("invalid element count (%d) while setting previously unknown unistate item \"%s\"\n",
  31.163 +					count, name);
  31.164 +			return -1;
  31.165 +		}
  31.166 +
  31.167 +		sidx = add_unistate(name, type);
  31.168 +	}
  31.169 +	set_unistate(sidx, val);
  31.170 +	return sidx;
  31.171 +}
  31.172 +
  31.173 +int set_unistate(const char *name, float *val, int count)
  31.174 +{
  31.175 +	int sidx = get_unistate_index(name);
  31.176 +	if(sidx < 0) {
  31.177 +		StType type = float_type(count);
  31.178 +		if(type == ST_UNKNOWN) {
  31.179 +			error_log("invalid element count (%d) while setting previously unknown unistate item \"%s\"\n",
  31.180 +					count, name);
  31.181 +			return -1;
  31.182 +		}
  31.183 +
  31.184 +		sidx = add_unistate(name, type);
  31.185 +	}
  31.186 +	set_unistate(sidx, val);
  31.187 +	return sidx;
  31.188 +}
  31.189 +
  31.190 +int set_unistate(const char *name, int val)
  31.191 +{
  31.192 +	int sidx = get_unistate_index(name);
  31.193 +	if(sidx < 0) {
  31.194 +		sidx = add_unistate(name, ST_INT);
  31.195 +	}
  31.196 +	set_unistate(sidx, val);
  31.197 +	return sidx;
  31.198 +}
  31.199 +
  31.200 +int set_unistate(const char *name, float val)
  31.201 +{
  31.202 +	int sidx = get_unistate_index(name);
  31.203 +	if(sidx < 0) {
  31.204 +		sidx = add_unistate(name, ST_FLOAT);
  31.205 +	}
  31.206 +	set_unistate(sidx, val);
  31.207 +	return sidx;
  31.208 +}
  31.209 +
  31.210 +int set_unistate(const char *name, const Vector2 &vec)
  31.211 +{
  31.212 +	int sidx = get_unistate_index(name);
  31.213 +	if(sidx < 0) {
  31.214 +		sidx = add_unistate(name, ST_FLOAT2);
  31.215 +	}
  31.216 +	set_unistate(sidx, vec);
  31.217 +	return sidx;
  31.218 +}
  31.219 +
  31.220 +int set_unistate(const char *name, const Vector3 &vec)
  31.221 +{
  31.222 +	int sidx = get_unistate_index(name);
  31.223 +	if(sidx < 0) {
  31.224 +		sidx = add_unistate(name, ST_FLOAT3);
  31.225 +	}
  31.226 +	set_unistate(sidx, vec);
  31.227 +	return sidx;
  31.228 +}
  31.229 +
  31.230 +int set_unistate(const char *name, const Vector4 &vec)
  31.231 +{
  31.232 +	int sidx = get_unistate_index(name);
  31.233 +	if(sidx < 0) {
  31.234 +		sidx = add_unistate(name, ST_FLOAT4);
  31.235 +	}
  31.236 +	set_unistate(sidx, vec);
  31.237 +	return sidx;
  31.238 +}
  31.239 +
  31.240 +int set_unistate(const char *name, const Matrix3x3 &mat)
  31.241 +{
  31.242 +	int sidx = get_unistate_index(name);
  31.243 +	if(sidx < 0) {
  31.244 +		sidx = add_unistate(name, ST_MATRIX3);
  31.245 +	}
  31.246 +	set_unistate(sidx, mat);
  31.247 +	return sidx;
  31.248 +}
  31.249 +
  31.250 +int set_unistate(const char *name, const Matrix4x4 &mat)
  31.251 +{
  31.252 +	int sidx = get_unistate_index(name);
  31.253 +	if(sidx < 0) {
  31.254 +		sidx = add_unistate(name, ST_MATRIX4);
  31.255 +	}
  31.256 +	set_unistate(sidx, mat);
  31.257 +	return sidx;
  31.258 +}
  31.259 +
  31.260 +
  31.261 +int get_unistate_int(int sidx)
  31.262 +{
  31.263 +	int val = 0;
  31.264 +	get_unistate(sidx, &val, 1);
  31.265 +	return val;
  31.266 +}
  31.267 +
  31.268 +float get_unistate_float(int sidx)
  31.269 +{
  31.270 +	float val = 0.0f;
  31.271 +	get_unistate(sidx, &val, 1);
  31.272 +	return val;
  31.273 +}
  31.274 +
  31.275 +Vector2 get_unistate_vec2(int sidx)
  31.276 +{
  31.277 +	float val[2] = {0.0f, 0.0f};
  31.278 +	get_unistate(sidx, val, 2);
  31.279 +	return Vector2(val[0], val[1]);
  31.280 +}
  31.281 +
  31.282 +Vector3 get_unistate_vec3(int sidx)
  31.283 +{
  31.284 +	float val[3] = {0.0f, 0.0f, 0.0f};
  31.285 +	get_unistate(sidx, val, 3);
  31.286 +	return Vector3(val[0], val[1], val[2]);
  31.287 +}
  31.288 +
  31.289 +Vector4 get_unistate_vec4(int sidx)
  31.290 +{
  31.291 +	float val[4] = {0.0f, 0.0f, 0.0f};
  31.292 +	get_unistate(sidx, val, 4);
  31.293 +	return Vector4(val[0], val[1], val[2], val[3]);
  31.294 +}
  31.295 +
  31.296 +Matrix3x3 get_unistate_mat3(int sidx)
  31.297 +{
  31.298 +	Matrix3x3 res;
  31.299 +	get_unistate(sidx, res.m[0], 9);
  31.300 +	return res;
  31.301 +}
  31.302 +
  31.303 +Matrix4x4 get_unistate_mat4(int sidx)
  31.304 +{
  31.305 +	Matrix4x4 res;
  31.306 +	get_unistate(sidx, res.m[0], 16);
  31.307 +	return res;
  31.308 +}
  31.309 +
  31.310 +
  31.311 +int get_unistate_int(const char *name)
  31.312 +{
  31.313 +	int sidx = get_unistate_index(name);
  31.314 +	if(sidx == -1) {
  31.315 +		return 0;
  31.316 +	}
  31.317 +	return get_unistate_int(sidx);
  31.318 +}
  31.319 +
  31.320 +float get_unistate_float(const char *name)
  31.321 +{
  31.322 +	int sidx = get_unistate_index(name);
  31.323 +	if(sidx == -1) {
  31.324 +		return 0.0f;
  31.325 +	}
  31.326 +	return get_unistate_float(sidx);
  31.327 +}
  31.328 +
  31.329 +Vector2 get_unistate_vec2(const char *name)
  31.330 +{
  31.331 +	int sidx = get_unistate_index(name);
  31.332 +	if(sidx == -1) {
  31.333 +		return Vector2();
  31.334 +	}
  31.335 +	return get_unistate_vec2(sidx);
  31.336 +}
  31.337 +
  31.338 +Vector3 get_unistate_vec3(const char *name)
  31.339 +{
  31.340 +	int sidx = get_unistate_index(name);
  31.341 +	if(sidx == -1) {
  31.342 +		return Vector3();
  31.343 +	}
  31.344 +	return get_unistate_vec3(sidx);
  31.345 +}
  31.346 +
  31.347 +Vector4 get_unistate_vec4(const char *name)
  31.348 +{
  31.349 +	int sidx = get_unistate_index(name);
  31.350 +	if(sidx == -1) {
  31.351 +		return Vector4();
  31.352 +	}
  31.353 +	return get_unistate_vec4(sidx);
  31.354 +}
  31.355 +
  31.356 +Matrix3x3 get_unistate_mat3(const char *name)
  31.357 +{
  31.358 +	int sidx = get_unistate_index(name);
  31.359 +	if(sidx == -1) {
  31.360 +		return Matrix3x3();
  31.361 +	}
  31.362 +	return get_unistate_mat3(sidx);
  31.363 +}
  31.364 +
  31.365 +Matrix4x4 get_unistate_mat4(const char *name)
  31.366 +{
  31.367 +	int sidx = get_unistate_index(name);
  31.368 +	if(sidx == -1) {
  31.369 +		return Matrix4x4();
  31.370 +	}
  31.371 +	return get_unistate_mat4(sidx);
  31.372 +}
  31.373 +
  31.374 +
  31.375 +void setup_unistate(const ShaderProg *sdr)
  31.376 +{
  31.377 +	if(!sdr) {
  31.378 +		if(!(sdr = ShaderProg::current)) {
  31.379 +			return;
  31.380 +		}
  31.381 +	}
  31.382 +
  31.383 +	sdr->setup_state_uniforms();
  31.384 +}
  31.385 +
  31.386 +bool setup_unistate(int sidx, const ShaderProg *sdr, int loc)
  31.387 +{
  31.388 +	if(loc < 0 || sidx < 0 || sidx >= (int)state.size()) {
  31.389 +		return false;
  31.390 +	}
  31.391 +
  31.392 +	CHECKGLERR;
  31.393 +	glUseProgram(sdr->get_id());
  31.394 +	CHECKGLERR;
  31.395 +
  31.396 +	switch(state[sidx].type) {
  31.397 +	case ST_INT:
  31.398 +		glUniform1iv(loc, 1, state[sidx].ival);
  31.399 +		break;
  31.400 +	case ST_INT2:
  31.401 +		glUniform2iv(loc, 1, state[sidx].ival);
  31.402 +		break;
  31.403 +	case ST_INT3:
  31.404 +		glUniform3iv(loc, 1, state[sidx].ival);
  31.405 +		break;
  31.406 +	case ST_INT4:
  31.407 +		glUniform4iv(loc, 1, state[sidx].ival);
  31.408 +		break;
  31.409 +
  31.410 +	case ST_FLOAT:
  31.411 +		glUniform1fv(loc, 1, state[sidx].fval);
  31.412 +		break;
  31.413 +	case ST_FLOAT2:
  31.414 +		glUniform2fv(loc, 1, state[sidx].fval);
  31.415 +		break;
  31.416 +	case ST_FLOAT3:
  31.417 +		glUniform3fv(loc, 1, state[sidx].fval);
  31.418 +		break;
  31.419 +	case ST_FLOAT4:
  31.420 +		glUniform4fv(loc, 1, state[sidx].fval);
  31.421 +		break;
  31.422 +
  31.423 +	case ST_MATRIX3:
  31.424 +#ifdef GL_ES_VERSION_2_0
  31.425 +		{
  31.426 +			float tmat[9], *ptr = tmat;
  31.427 +			for(int i=0; i<3; i++) {
  31.428 +				for(int j=0; j<3; j++) {
  31.429 +					*ptr++ = state[sidx].fval[j * 3 + i];
  31.430 +				}
  31.431 +			}
  31.432 +			glUniformMatrix3fv(loc, 1, GL_FALSE, tmat);
  31.433 +		}
  31.434 +#else
  31.435 +		glUniformMatrix3fv(loc, 1, state[sidx].transpose, state[sidx].fval);
  31.436 +#endif
  31.437 +		break;
  31.438 +
  31.439 +	case ST_MATRIX4:
  31.440 +#ifdef GL_ES_VERSION_2_0
  31.441 +		{
  31.442 +			float tmat[16], *ptr = tmat;
  31.443 +			for(int i=0; i<4; i++) {
  31.444 +				for(int j=0; j<4; j++) {
  31.445 +					*ptr++ = state[sidx].fval[j * 4 + i];
  31.446 +				}
  31.447 +			}
  31.448 +			glUniformMatrix4fv(loc, 1, GL_FALSE, tmat);
  31.449 +		}
  31.450 +#else
  31.451 +		glUniformMatrix4fv(loc, 1, state[sidx].transpose, state[sidx].fval);
  31.452 +#endif
  31.453 +		break;
  31.454 +
  31.455 +	default:
  31.456 +		return false;
  31.457 +	}
  31.458 +
  31.459 +	CHECKGLERR;
  31.460 +	return true;
  31.461 +}
  31.462 +
  31.463 +bool setup_unistate(const char *name, const ShaderProg *sdr)
  31.464 +{
  31.465 +	int loc = sdr->get_uniform_location(name);
  31.466 +	if(loc == -1) {
  31.467 +		return false;
  31.468 +	}
  31.469 +	return setup_unistate(get_unistate_index(name), sdr, loc);
  31.470 +}
  31.471 +
  31.472 +void set_world_matrix(const Matrix4x4 &mat)
  31.473 +{
  31.474 +	static int sidx = -1, sidx_transp, sidx_mat3;
  31.475 +
  31.476 +	if(sidx == -1) {
  31.477 +		sidx = add_unistate("st_world_matrix", ST_MATRIX4);
  31.478 +		sidx_mat3 = add_unistate("st_world_matrix3", ST_MATRIX3);
  31.479 +		sidx_transp = add_unistate("st_world_matrix_transpose", ST_MATRIX4);
  31.480 +	}
  31.481 +
  31.482 +	set_unistate(sidx, mat);
  31.483 +	set_unistate(sidx_mat3, Matrix3x3(mat));
  31.484 +	set_unistate(sidx_transp, mat[0]);	// by using the float* variant, we unset the transpose flag
  31.485 +}
  31.486 +
  31.487 +void set_view_matrix(const Matrix4x4 &mat)
  31.488 +{
  31.489 +	static int sidx = -1, sidx_transp, sidx_mat3;
  31.490 +
  31.491 +	if(sidx == -1) {
  31.492 +		sidx = add_unistate("st_view_matrix", ST_MATRIX4);
  31.493 +		sidx_mat3 = add_unistate("st_view_matrix3", ST_MATRIX3);
  31.494 +		sidx_transp = add_unistate("st_view_matrix_transpose", ST_MATRIX4);
  31.495 +	}
  31.496 +
  31.497 +	set_unistate(sidx, mat);
  31.498 +	set_unistate(sidx_mat3, Matrix3x3(mat));
  31.499 +	set_unistate(sidx_transp, mat[0]);	// by using the float* variant, we unset the transpose flag
  31.500 +}
  31.501 +
  31.502 +void set_projection_matrix(const Matrix4x4 &mat)
  31.503 +{
  31.504 +	static int sidx = -1;
  31.505 +
  31.506 +	if(sidx == -1) {
  31.507 +		sidx = add_unistate("st_proj_matrix", ST_MATRIX4);
  31.508 +	}
  31.509 +
  31.510 +	set_unistate(sidx, mat);
  31.511 +}
  31.512 +
  31.513 +void set_texture_matrix(const Matrix4x4 &mat)
  31.514 +{
  31.515 +	static int sidx = -1;
  31.516 +
  31.517 +	if(sidx == -1) {
  31.518 +		sidx = add_unistate("st_tex_matrix", ST_MATRIX4);
  31.519 +	}
  31.520 +
  31.521 +	set_unistate(sidx, mat);
  31.522 +}
  31.523 +
  31.524 +Matrix4x4 get_world_matrix()
  31.525 +{
  31.526 +	static int sidx = -1;
  31.527 +
  31.528 +	if(sidx == -1) {
  31.529 +		if((sidx = get_unistate_index("st_world_matrix")) == -1) {
  31.530 +			return Matrix4x4();
  31.531 +		}
  31.532 +	}
  31.533 +	return get_unistate_mat4(sidx);
  31.534 +}
  31.535 +
  31.536 +Matrix4x4 get_view_matrix()
  31.537 +{
  31.538 +	static int sidx = -1;
  31.539 +
  31.540 +	if(sidx == -1) {
  31.541 +		if((sidx = get_unistate_index("st_view_matrix")) == -1) {
  31.542 +			return Matrix4x4();
  31.543 +		}
  31.544 +	}
  31.545 +	return get_unistate_mat4(sidx);
  31.546 +}
  31.547 +
  31.548 +Matrix4x4 get_projection_matrix()
  31.549 +{
  31.550 +	static int sidx = -1;
  31.551 +
  31.552 +	if(sidx == -1) {
  31.553 +		if((sidx = get_unistate_index("st_proj_matrix")) == -1) {
  31.554 +			return Matrix4x4();
  31.555 +		}
  31.556 +	}
  31.557 +	return get_unistate_mat4(sidx);
  31.558 +}
  31.559 +
  31.560 +Matrix4x4 get_texture_matrix()
  31.561 +{
  31.562 +	static int sidx = -1;
  31.563 +
  31.564 +	if(sidx == -1) {
  31.565 +		if((sidx = get_unistate_index("st_tex_matrix")) == -1) {
  31.566 +			return Matrix4x4();
  31.567 +		}
  31.568 +	}
  31.569 +	return get_unistate_mat4(sidx);
  31.570 +}
  31.571 +
  31.572 +void setup_gl_matrices()
  31.573 +{
  31.574 +#ifdef USE_OLDGL
  31.575 +	Matrix4x4 modelview = get_world_matrix() * get_view_matrix();
  31.576 +	Matrix4x4 proj = get_projection_matrix();
  31.577 +	Matrix4x4 tex = get_texture_matrix();
  31.578 +
  31.579 +	glMatrixMode(GL_TEXTURE);
  31.580 +	glLoadTransposeMatrixf(tex[0]);
  31.581 +	glMatrixMode(GL_PROJECTION);
  31.582 +	glLoadTransposeMatrixf(proj[0]);
  31.583 +	glMatrixMode(GL_MODELVIEW);
  31.584 +	glLoadTransposeMatrixf(modelview[0]);
  31.585 +#endif
  31.586 +}
  31.587 +
  31.588 +static const char *typestr(StType type)
  31.589 +{
  31.590 +	switch(type) {
  31.591 +	case ST_INT:
  31.592 +		return "int";
  31.593 +	case ST_INT2:
  31.594 +		return "ivec2";
  31.595 +	case ST_INT3:
  31.596 +		return "ivec3";
  31.597 +	case ST_INT4:
  31.598 +		return "ivec4";
  31.599 +	case ST_FLOAT:
  31.600 +		return "float";
  31.601 +	case ST_FLOAT2:
  31.602 +		return "vec2";
  31.603 +	case ST_FLOAT3:
  31.604 +		return "vec3";
  31.605 +	case ST_FLOAT4:
  31.606 +		return "vec4";
  31.607 +	case ST_MATRIX3:
  31.608 +		return "mat3";
  31.609 +	case ST_MATRIX4:
  31.610 +		return "mat4";
  31.611 +
  31.612 +	default:
  31.613 +		break;
  31.614 +	}
  31.615 +	return "<unknown>";
  31.616 +}
  31.617 +
  31.618 +static int type_nelem(StType type)
  31.619 +{
  31.620 +	switch(type) {
  31.621 +	case ST_INT:
  31.622 +	case ST_FLOAT:
  31.623 +		return 1;
  31.624 +	case ST_INT2:
  31.625 +	case ST_FLOAT2:
  31.626 +		return 2;
  31.627 +	case ST_INT3:
  31.628 +	case ST_FLOAT3:
  31.629 +		return 3;
  31.630 +	case ST_INT4:
  31.631 +	case ST_FLOAT4:
  31.632 +		return 4;
  31.633 +	case ST_MATRIX3:
  31.634 +		return 9;
  31.635 +	case ST_MATRIX4:
  31.636 +		return 16;
  31.637 +
  31.638 +	default:
  31.639 +		break;
  31.640 +	}
  31.641 +
  31.642 +	return 0;
  31.643 +}
  31.644 +
  31.645 +static StType float_type(int elem)
  31.646 +{
  31.647 +	switch(elem) {
  31.648 +	case 1:
  31.649 +		return ST_FLOAT;
  31.650 +	case 2:
  31.651 +		return ST_FLOAT2;
  31.652 +	case 3:
  31.653 +		return ST_FLOAT3;
  31.654 +	case 4:
  31.655 +		return ST_FLOAT4;
  31.656 +	case 9:
  31.657 +		return ST_MATRIX3;
  31.658 +	case 16:
  31.659 +		return ST_MATRIX4;
  31.660 +	default:
  31.661 +		break;
  31.662 +	}
  31.663 +	return ST_UNKNOWN;
  31.664 +}
  31.665 +
  31.666 +static StType int_type(int elem)
  31.667 +{
  31.668 +	switch(elem) {
  31.669 +	case 1:
  31.670 +		return ST_INT;
  31.671 +	case 2:
  31.672 +		return ST_INT2;
  31.673 +	case 3:
  31.674 +		return ST_INT3;
  31.675 +	case 4:
  31.676 +		return ST_INT4;
  31.677 +	default:
  31.678 +		break;
  31.679 +	}
  31.680 +	return ST_UNKNOWN;
  31.681 +}
    32.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    32.2 +++ b/src/unistate.h	Sun Aug 24 09:41:24 2014 +0300
    32.3 @@ -0,0 +1,104 @@
    32.4 +#ifndef UNISTATE_H_
    32.5 +#define UNISTATE_H_
    32.6 +
    32.7 +#include "vmath/vmath.h"
    32.8 +
    32.9 +class ShaderProg;
   32.10 +
   32.11 +enum StType {
   32.12 +	ST_UNKNOWN,
   32.13 +	ST_INT,	ST_INT2, ST_INT3, ST_INT4,
   32.14 +	ST_FLOAT, ST_FLOAT2, ST_FLOAT3, ST_FLOAT4,
   32.15 +	ST_MATRIX3, ST_MATRIX4
   32.16 +};
   32.17 +
   32.18 +int add_unistate(const char *name, StType type);
   32.19 +int get_unistate_index(const char *name);
   32.20 +
   32.21 +/** set the uniform state identified by \param sidx by copying
   32.22 + * a number of elements from \param val. If \param count is 0
   32.23 + * then it's automatically set based on the type of this state item.
   32.24 + * @{ */
   32.25 +void set_unistate(int sidx, const int *val, int count = 0);
   32.26 +void set_unistate(int sidx, const float *val, int count = 0);
   32.27 +/// @}
   32.28 +
   32.29 +/** get the uniform state identified by \param sidx by copying
   32.30 + * a number of elements into \param val. If \param count is 0
   32.31 + * then it's automatically set based on the type of this state item.
   32.32 + * @{ */
   32.33 +void get_unistate(int sidx, int *val, int count = 0);
   32.34 +void get_unistate(int sidx, float *val, int count = 0);
   32.35 +/// @}
   32.36 +
   32.37 +/// convenience versions of set_unistate @{
   32.38 +void set_unistate(int sidx, int val);
   32.39 +void set_unistate(int sidx, float val);
   32.40 +void set_unistate(int sidx, const Vector2 &vec);
   32.41 +void set_unistate(int sidx, const Vector3 &vec);
   32.42 +void set_unistate(int sidx, const Vector4 &vec);
   32.43 +void set_unistate(int sidx, const Matrix3x3 &mat);
   32.44 +void set_unistate(int sidx, const Matrix4x4 &mat);
   32.45 +/// @}
   32.46 +
   32.47 +/** convenience functions for setting the uniform state by name.
   32.48 + * if the name cannot be found in the current set of uniform state
   32.49 + * items, a new one is created with a type derived from the variant
   32.50 + * of the function that was called (which might not be what you want).
   32.51 + * The index of the state item is returned.
   32.52 + * @{ */
   32.53 +int set_unistate(const char *name, int *val, int count = 0);
   32.54 +int set_unistate(const char *name, float *val, int count = 0);
   32.55 +int set_unistate(const char *name, int val);
   32.56 +int set_unistate(const char *name, float val);
   32.57 +int set_unistate(const char *name, const Vector2 &vec);
   32.58 +int set_unistate(const char *name, const Vector3 &vec);
   32.59 +int set_unistate(const char *name, const Vector4 &vec);
   32.60 +int set_unistate(const char *name, const Matrix3x3 &mat);
   32.61 +int set_unistate(const char *name, const Matrix4x4 &mat);
   32.62 +/// @}
   32.63 +
   32.64 +/// convenience versions of get_unistate @{
   32.65 +int get_unistate_int(int sidx);
   32.66 +float get_unistate_float(int sidx);
   32.67 +Vector2 get_unistate_vec2(int sidx);
   32.68 +Vector3 get_unistate_vec3(int sidx);
   32.69 +Vector4 get_unistate_vec4(int sidx);
   32.70 +Matrix3x3 get_unistate_mat3(int sidx);
   32.71 +Matrix4x4 get_unistate_mat4(int sidx);
   32.72 +/// @}
   32.73 +
   32.74 +/// convenience versions of get_unistate for getting the uniform state by name @{
   32.75 +int get_unistate_int(const char *name);
   32.76 +float get_unistate_float(const char *name);
   32.77 +Vector2 get_unistate_vec2(const char *name);
   32.78 +Vector3 get_unistate_vec3(const char *name);
   32.79 +Vector4 get_unistate_vec4(const char *name);
   32.80 +Matrix3x3 get_unistate_mat3(const char *name);
   32.81 +Matrix4x4 get_unistate_mat4(const char *name);
   32.82 +/// @}
   32.83 +
   32.84 +/** Prepare for rendering by setting up all the state uniforms in the shader sdr.
   32.85 + * If sdr is null, then use the "current" shader as per ShaderProg::current
   32.86 + */
   32.87 +void setup_unistate(const ShaderProg *sdr = 0);
   32.88 +
   32.89 +bool setup_unistate(int sidx, const ShaderProg *sdr, int loc);
   32.90 +bool setup_unistate(const char *name, const ShaderProg *sdr);
   32.91 +
   32.92 +// special functions for setting the rendering pipeline matrices
   32.93 +void set_world_matrix(const Matrix4x4 &mat);
   32.94 +void set_view_matrix(const Matrix4x4 &mat);
   32.95 +void set_projection_matrix(const Matrix4x4 &mat);
   32.96 +void set_texture_matrix(const Matrix4x4 &mat);
   32.97 +
   32.98 +Matrix4x4 get_world_matrix();
   32.99 +Matrix4x4 get_view_matrix();
  32.100 +Matrix4x4 get_projection_matrix();
  32.101 +Matrix4x4 get_texture_matrix();
  32.102 +
  32.103 +void setup_gl_matrices();	// this shouldn't be needed in the final code
  32.104 +
  32.105 +// TODO should do a matrix stack at some point ...
  32.106 +
  32.107 +#endif	// UNISTATE_H_
    33.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    33.2 +++ b/src/xform_node.cc	Sun Aug 24 09:41:24 2014 +0300
    33.3 @@ -0,0 +1,461 @@
    33.4 +#include <assert.h>
    33.5 +#include <algorithm>
    33.6 +#include "xform_node.h"
    33.7 +#include "anim/anim.h"
    33.8 +#include "anim/track.h"
    33.9 +
   33.10 +static inline anm_interpolator track_interpolator(Interp in);
   33.11 +static inline anm_extrapolator track_extrapolator(Extrap ex);
   33.12 +
   33.13 +XFormNode::XFormNode()
   33.14 +{
   33.15 +	anm = new anm_node;
   33.16 +	anm_init_node(anm);
   33.17 +	parent = 0;
   33.18 +
   33.19 +	// TODO read them from anm to get the correct initial values
   33.20 +	interp = INTERP_LINEAR;
   33.21 +	extrap = EXTRAP_EXTEND;
   33.22 +}
   33.23 +
   33.24 +XFormNode::~XFormNode()
   33.25 +{
   33.26 +	anm_destroy_node(anm);
   33.27 +	delete anm;
   33.28 +}
   33.29 +
   33.30 +void XFormNode::set_name(const char *name)
   33.31 +{
   33.32 +	anm_set_node_name(anm, name);
   33.33 +}
   33.34 +
   33.35 +const char *XFormNode::get_name() const
   33.36 +{
   33.37 +	return anm_get_node_name(anm);
   33.38 +}
   33.39 +
   33.40 +void XFormNode::set_interpolator(Interp in)
   33.41 +{
   33.42 +	anm_set_interpolator(anm, track_interpolator(in));
   33.43 +	interp = in;
   33.44 +}
   33.45 +
   33.46 +Interp XFormNode::get_interpolator() const
   33.47 +{
   33.48 +	return interp;
   33.49 +}
   33.50 +
   33.51 +void XFormNode::set_extrapolator(Extrap ex)
   33.52 +{
   33.53 +	anm_set_extrapolator(anm, track_extrapolator(ex));
   33.54 +	extrap = ex;
   33.55 +}
   33.56 +
   33.57 +Extrap XFormNode::get_extrapolator() const
   33.58 +{
   33.59 +	return extrap;
   33.60 +}
   33.61 +
   33.62 +XFormNode *XFormNode::get_parent()
   33.63 +{
   33.64 +	return parent;
   33.65 +}
   33.66 +
   33.67 +const XFormNode *XFormNode::get_parent() const
   33.68 +{
   33.69 +	return parent;
   33.70 +}
   33.71 +
   33.72 +void XFormNode::add_child(XFormNode *child)
   33.73 +{
   33.74 +	children.push_back(child);
   33.75 +	anm_link_node(anm, child->anm);
   33.76 +	child->parent = this;
   33.77 +}
   33.78 +
   33.79 +void XFormNode::remove_child(XFormNode *child)
   33.80 +{
   33.81 +	std::vector<XFormNode*>::iterator it;
   33.82 +	it = std::find(children.begin(), children.end(), child);
   33.83 +	if(it != children.end()) {
   33.84 +		children.erase(it);
   33.85 +		anm_unlink_node(anm, child->anm);
   33.86 +
   33.87 +		if(child->parent == this) {
   33.88 +			child->parent = 0;
   33.89 +		}
   33.90 +	}
   33.91 +}
   33.92 +
   33.93 +int XFormNode::get_children_count() const
   33.94 +{
   33.95 +	return (int)children.size();
   33.96 +}
   33.97 +
   33.98 +XFormNode *XFormNode::get_child(int idx)
   33.99 +{
  33.100 +	if(idx >= 0 && idx < get_children_count()) {
  33.101 +		return children[idx];
  33.102 +	}
  33.103 +	return 0;
  33.104 +}
  33.105 +
  33.106 +const XFormNode *XFormNode::get_child(int idx) const
  33.107 +{
  33.108 +	if(idx >= 0 && idx < get_children_count()) {
  33.109 +		return children[idx];
  33.110 +	}
  33.111 +	return 0;
  33.112 +}
  33.113 +
  33.114 +
  33.115 +void XFormNode::use_animation(int idx)
  33.116 +{
  33.117 +	if(idx >= 0) {
  33.118 +		anm_use_animation(anm, idx);
  33.119 +	}
  33.120 +}
  33.121 +
  33.122 +void XFormNode::use_animation(const char *name)
  33.123 +{
  33.124 +	anm_use_animation(anm, anm_find_animation(anm, name));
  33.125 +}
  33.126 +
  33.127 +void XFormNode::use_animation(int aidx, int bidx, float t)
  33.128 +{
  33.129 +	anm_use_animations(anm, aidx, bidx, t);
  33.130 +}
  33.131 +
  33.132 +void XFormNode::use_animation(const char *aname, const char *bname, float t)
  33.133 +{
  33.134 +	int aidx = anm_find_animation(anm, aname);
  33.135 +	int bidx = anm_find_animation(anm, bname);
  33.136 +
  33.137 +	if(aidx == -1) {
  33.138 +		use_animation(bidx);
  33.139 +	}
  33.140 +	if(bidx == -1) {
  33.141 +		use_animation(aidx);
  33.142 +	}
  33.143 +	anm_use_animations(anm, aidx, bidx, t);
  33.144 +}
  33.145 +
  33.146 +int XFormNode::get_active_animation_index(int which) const
  33.147 +{
  33.148 +	return anm_get_active_animation_index(anm, which);
  33.149 +}
  33.150 +
  33.151 +float XFormNode::get_active_animation_mix() const
  33.152 +{
  33.153 +	return anm_get_active_animation_mix(anm);
  33.154 +}
  33.155 +
  33.156 +int XFormNode::get_animation_count() const
  33.157 +{
  33.158 +	return anm_get_animation_count(anm);
  33.159 +}
  33.160 +
  33.161 +void XFormNode::add_animation(const char *name)
  33.162 +{
  33.163 +	int idx = get_animation_count();
  33.164 +
  33.165 +	anm_add_animation(anm);
  33.166 +	use_animation(idx);
  33.167 +
  33.168 +	if(name) {
  33.169 +		set_animation_name(name);
  33.170 +	}
  33.171 +}
  33.172 +
  33.173 +void XFormNode::set_animation_name(const char *name)
  33.174 +{
  33.175 +	anm_set_active_animation_name(anm, name);
  33.176 +}
  33.177 +
  33.178 +const char *XFormNode::get_animation_name() const
  33.179 +{
  33.180 +	return anm_get_active_animation_name(anm);
  33.181 +}
  33.182 +
  33.183 +
  33.184 +
  33.185 +void XFormNode::set_position(const Vector3 &pos, long tmsec)
  33.186 +{
  33.187 +	anm_set_position(anm, v3_cons(pos.x, pos.y, pos.z), ANM_MSEC2TM(tmsec));
  33.188 +}
  33.189 +
  33.190 +Vector3 XFormNode::get_node_position(long tmsec) const
  33.191 +{
  33.192 +	vec3_t p = anm_get_node_position(anm, ANM_MSEC2TM(tmsec));
  33.193 +	return Vector3(p.x, p.y, p.z);
  33.194 +}
  33.195 +
  33.196 +void XFormNode::set_rotation(const Quaternion &quat, long tmsec)
  33.197 +{
  33.198 +	anm_set_rotation(anm, quat_cons(quat.s, quat.v.x, quat.v.y, quat.v.z), ANM_MSEC2TM(tmsec));
  33.199 +}
  33.200 +
  33.201 +Quaternion XFormNode::get_node_rotation(long tmsec) const
  33.202 +{
  33.203 +	quat_t q = anm_get_node_rotation(anm, ANM_MSEC2TM(tmsec));
  33.204 +	return Quaternion(q.w, q.x, q.y, q.z);
  33.205 +}
  33.206 +
  33.207 +void XFormNode::set_scaling(const Vector3 &pos, long tmsec)
  33.208 +{
  33.209 +	anm_set_scaling(anm, v3_cons(pos.x, pos.y, pos.z), ANM_MSEC2TM(tmsec));
  33.210 +}
  33.211 +
  33.212 +Vector3 XFormNode::get_node_scaling(long tmsec) const
  33.213 +{
  33.214 +	vec3_t s = anm_get_node_scaling(anm, ANM_MSEC2TM(tmsec));
  33.215 +	return Vector3(s.x, s.y, s.z);
  33.216 +}
  33.217 +
  33.218 +// these take hierarchy into account
  33.219 +Vector3 XFormNode::get_position(long tmsec) const
  33.220 +{
  33.221 +	vec3_t v = anm_get_position(anm, ANM_MSEC2TM(tmsec));
  33.222 +	return Vector3(v.x, v.y, v.z);
  33.223 +}
  33.224 +
  33.225 +Quaternion XFormNode::get_rotation(long tmsec) const
  33.226 +{
  33.227 +	quat_t q = anm_get_rotation(anm, tmsec);
  33.228 +	return Quaternion(q.w, q.x, q.y, q.z);
  33.229 +}
  33.230 +
  33.231 +Vector3 XFormNode::get_scaling(long tmsec) const
  33.232 +{
  33.233 +	vec3_t v = anm_get_scaling(anm, ANM_MSEC2TM(tmsec));
  33.234 +	return Vector3(v.x, v.y, v.z);
  33.235 +}
  33.236 +
  33.237 +void XFormNode::set_pivot(const Vector3 &pivot)
  33.238 +{
  33.239 +	anm_set_pivot(anm, v3_cons(pivot.x, pivot.y, pivot.z));
  33.240 +}
  33.241 +
  33.242 +Vector3 XFormNode::get_pivot() const
  33.243 +{
  33.244 +	vec3_t p = anm_get_pivot(anm);
  33.245 +	return Vector3(p.x, p.y, p.z);
  33.246 +}
  33.247 +
  33.248 +void XFormNode::set_local_matrix(const Matrix4x4 &mat)
  33.249 +{
  33.250 +	local_matrix = mat;
  33.251 +}
  33.252 +
  33.253 +const Matrix4x4 &XFormNode::get_local_matrix() const
  33.254 +{
  33.255 +	return local_matrix;
  33.256 +}
  33.257 +
  33.258 +void XFormNode::set_bone_matrix(const Matrix4x4 &bmat)
  33.259 +{
  33.260 +	bone_matrix = bmat;
  33.261 +}
  33.262 +
  33.263 +const Matrix4x4 &XFormNode::get_bone_matrix() const
  33.264 +{
  33.265 +	return bone_matrix;
  33.266 +}
  33.267 +
  33.268 +#define FOO
  33.269 +
  33.270 +void XFormNode::get_node_xform(long tmsec, Matrix4x4 *mat, Matrix4x4 *inv_mat) const
  33.271 +{
  33.272 +	anm_time_t tm = ANM_MSEC2TM(tmsec);
  33.273 +
  33.274 +	if(mat) {
  33.275 +		anm_get_node_matrix(anm, (scalar_t(*)[4])mat, tm);
  33.276 +#ifdef FOO
  33.277 +		*mat = local_matrix * *mat;
  33.278 +#else
  33.279 +		*mat = *mat * local_matrix;
  33.280 +#endif
  33.281 +	}
  33.282 +	if(inv_mat) {
  33.283 +		anm_get_inv_matrix(anm, (scalar_t(*)[4])inv_mat, tm);
  33.284 +	}
  33.285 +}
  33.286 +
  33.287 +void XFormNode::get_xform(long tmsec, Matrix4x4 *mat, Matrix4x4 *inv_mat) const
  33.288 +{
  33.289 +	anm_time_t tm = ANM_MSEC2TM(tmsec);
  33.290 +
  33.291 +	if(mat) {
  33.292 +		anm_get_matrix(anm, (scalar_t(*)[4])mat, tm);
  33.293 +#ifdef FOO
  33.294 +		*mat = local_matrix * *mat;
  33.295 +#else
  33.296 +		*mat = *mat * local_matrix;
  33.297 +#endif
  33.298 +	}
  33.299 +	if(inv_mat) {
  33.300 +		anm_get_inv_matrix(anm, (scalar_t(*)[4])inv_mat, tm);
  33.301 +	}
  33.302 +}
  33.303 +
  33.304 +
  33.305 +// ---- Track ----
  33.306 +
  33.307 +Track::Track()
  33.308 +{
  33.309 +	trk = new anm_track;
  33.310 +	anm_init_track(trk);
  33.311 +}
  33.312 +
  33.313 +Track::~Track()
  33.314 +{
  33.315 +	anm_destroy_track(trk);
  33.316 +	delete trk;
  33.317 +}
  33.318 +
  33.319 +Track::Track(const Track &rhs)
  33.320 +{
  33.321 +	trk = new anm_track;
  33.322 +	anm_init_track(trk);
  33.323 +	anm_copy_track(trk, rhs.trk);
  33.324 +	interp = rhs.interp;
  33.325 +	extrap = rhs.extrap;
  33.326 +}
  33.327 +
  33.328 +Track &Track::operator =(const Track &rhs)
  33.329 +{
  33.330 +	if(&rhs == this) {
  33.331 +		return *this;
  33.332 +	}
  33.333 +
  33.334 +	anm_copy_track(trk, rhs.trk);
  33.335 +	interp = rhs.interp;
  33.336 +	extrap = rhs.extrap;
  33.337 +	return *this;
  33.338 +}
  33.339 +
  33.340 +
  33.341 +void Track::set_interpolator(Interp in)
  33.342 +{
  33.343 +	anm_set_track_interpolator(trk, track_interpolator(in));
  33.344 +	interp = in;
  33.345 +}
  33.346 +
  33.347 +Interp Track::get_interpolator() const
  33.348 +{
  33.349 +	return interp;
  33.350 +}
  33.351 +
  33.352 +void Track::set_extrapolator(Extrap ex)
  33.353 +{
  33.354 +	anm_set_track_extrapolator(trk, track_extrapolator(ex));
  33.355 +	extrap = ex;
  33.356 +}
  33.357 +
  33.358 +Extrap Track::get_extrapolator() const
  33.359 +{
  33.360 +	return extrap;
  33.361 +}
  33.362 +
  33.363 +void Track::set_default(double def)
  33.364 +{
  33.365 +	anm_set_track_default(trk, def);
  33.366 +}
  33.367 +
  33.368 +void Track::set_value(float val, long tmsec)
  33.369 +{
  33.370 +	anm_set_value(trk, ANM_MSEC2TM(tmsec), val);
  33.371 +}
  33.372 +
  33.373 +float Track::get_value(long tmsec) const
  33.374 +{
  33.375 +	return anm_get_value(trk, ANM_MSEC2TM(tmsec));
  33.376 +}
  33.377 +
  33.378 +float Track::operator ()(long tmsec) const
  33.379 +{
  33.380 +	return anm_get_value(trk, ANM_MSEC2TM(tmsec));
  33.381 +}
  33.382 +
  33.383 +
  33.384 +// ---- Track3 ----
  33.385 +
  33.386 +void Track3::set_interpolator(Interp in)
  33.387 +{
  33.388 +	for(int i=0; i<3; i++) {
  33.389 +		track[i].set_interpolator(in);
  33.390 +	}
  33.391 +}
  33.392 +
  33.393 +Interp Track3::get_interpolator() const
  33.394 +{
  33.395 +	return track[0].get_interpolator();
  33.396 +}
  33.397 +
  33.398 +void Track3::set_extrapolator(Extrap ex)
  33.399 +{
  33.400 +	for(int i=0; i<3; i++) {
  33.401 +		track[i].set_extrapolator(ex);
  33.402 +	}
  33.403 +}
  33.404 +
  33.405 +Extrap Track3::get_extrapolator() const
  33.406 +{
  33.407 +	return track[0].get_extrapolator();
  33.408 +}
  33.409 +
  33.410 +void Track3::set_default(const Vector3 &def)
  33.411 +{
  33.412 +	for(int i=0; i<3; i++) {
  33.413 +		track[i].set_default(def[i]);
  33.414 +	}
  33.415 +}
  33.416 +
  33.417 +void Track3::set_value(const Vector3 &val, long tmsec)
  33.418 +{
  33.419 +	for(int i=0; i<3; i++) {
  33.420 +		track[i].set_value(val[i], tmsec);
  33.421 +	}
  33.422 +}
  33.423 +
  33.424 +Vector3 Track3::get_value(long tmsec) const
  33.425 +{
  33.426 +	return Vector3(track[0](tmsec), track[1](tmsec), track[2](tmsec));
  33.427 +}
  33.428 +
  33.429 +Vector3 Track3::operator ()(long tmsec) const
  33.430 +{
  33.431 +	return Vector3(track[0](tmsec), track[1](tmsec), track[2](tmsec));
  33.432 +}
  33.433 +
  33.434 +
  33.435 +static inline anm_interpolator track_interpolator(Interp in)
  33.436 +{
  33.437 +	switch(in) {
  33.438 +	case INTERP_STEP:
  33.439 +		return ANM_INTERP_STEP;
  33.440 +	case INTERP_LINEAR:
  33.441 +		return ANM_INTERP_LINEAR;
  33.442 +	case INTERP_CUBIC:
  33.443 +		return ANM_INTERP_CUBIC;
  33.444 +	}
  33.445 +
  33.446 +	assert(0);
  33.447 +	return ANM_INTERP_STEP;
  33.448 +}
  33.449 +
  33.450 +static inline anm_extrapolator track_extrapolator(Extrap ex)
  33.451 +{
  33.452 +	switch(ex) {
  33.453 +	case EXTRAP_EXTEND:
  33.454 +		return ANM_EXTRAP_EXTEND;
  33.455 +	case EXTRAP_CLAMP:
  33.456 +		return ANM_EXTRAP_CLAMP;
  33.457 +	case EXTRAP_REPEAT:
  33.458 +		return ANM_EXTRAP_REPEAT;
  33.459 +	}
  33.460 +
  33.461 +	assert(0);
  33.462 +	return ANM_EXTRAP_EXTEND;
  33.463 +}
  33.464 +
    34.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
    34.2 +++ b/src/xform_node.h	Sun Aug 24 09:41:24 2014 +0300
    34.3 @@ -0,0 +1,154 @@
    34.4 +#ifndef GOATGFX_XFORM_NODE_H_
    34.5 +#define GOATGFX_XFORM_NODE_H_
    34.6 +
    34.7 +#include <vector>
    34.8 +#include "vmath/vector.h"
    34.9 +#include "vmath/quat.h"
   34.10 +#include "vmath/matrix.h"
   34.11 +
   34.12 +
   34.13 +struct anm_node;
   34.14 +struct anm_track;
   34.15 +
   34.16 +enum Interp { INTERP_STEP, INTERP_LINEAR, INTERP_CUBIC };
   34.17 +enum Extrap { EXTRAP_EXTEND, EXTRAP_CLAMP, EXTRAP_REPEAT };
   34.18 +
   34.19 +// NOTE: all time arguments are milliseconds
   34.20 +
   34.21 +class XFormNode {
   34.22 +private:
   34.23 +	struct anm_node *anm;
   34.24 +	std::vector<XFormNode*> children;
   34.25 +	XFormNode *parent;
   34.26 +
   34.27 +	Interp interp;
   34.28 +	Extrap extrap;
   34.29 +
   34.30 +	Matrix4x4 local_matrix;
   34.31 +	Matrix4x4 bone_matrix;
   34.32 +
   34.33 +	XFormNode(const XFormNode &node) {}
   34.34 +	XFormNode &operator =(const XFormNode &node) { return *this; }
   34.35 +
   34.36 +public:
   34.37 +	XFormNode();
   34.38 +	virtual ~XFormNode();
   34.39 +
   34.40 +	virtual void set_name(const char *name);
   34.41 +	virtual const char *get_name() const;
   34.42 +
   34.43 +	virtual void set_interpolator(Interp in);
   34.44 +	virtual Interp get_interpolator() const;
   34.45 +	virtual void set_extrapolator(Extrap ex);
   34.46 +	virtual Extrap get_extrapolator() const;
   34.47 +
   34.48 +	virtual XFormNode *get_parent();
   34.49 +	virtual const XFormNode *get_parent() const;
   34.50 +
   34.51 +	// children management
   34.52 +	virtual void add_child(XFormNode *child);
   34.53 +	virtual void remove_child(XFormNode *child);
   34.54 +
   34.55 +	virtual int get_children_count() const;
   34.56 +	virtual XFormNode *get_child(int idx);
   34.57 +	virtual const XFormNode *get_child(int idx) const;
   34.58 +
   34.59 +
   34.60 +	virtual void use_animation(int idx);
   34.61 +	virtual void use_animation(const char *name);
   34.62 +	virtual void use_animation(int aidx, int bidx, float t);
   34.63 +	virtual void use_animation(const char *aname, const char *bname, float t);
   34.64 +
   34.65 +	virtual int get_active_animation_index(int which = 0) const;
   34.66 +	virtual float get_active_animation_mix() const;
   34.67 +
   34.68 +	virtual int get_animation_count() const;
   34.69 +
   34.70 +	// add a new empty animation slot (recursive)
   34.71 +	virtual void add_animation(const char *name = 0);
   34.72 +
   34.73 +	// set/get the current animation name (set is recursive)
   34.74 +	virtual void set_animation_name(const char *name);
   34.75 +	virtual const char *get_animation_name() const;
   34.76 +
   34.77 +
   34.78 +	virtual void set_position(const Vector3 &pos, long tmsec = 0);
   34.79 +	virtual Vector3 get_node_position(long tmsec = 0) const;
   34.80 +
   34.81 +	virtual void set_rotation(const Quaternion &quat, long tmsec = 0);
   34.82 +	virtual Quaternion get_node_rotation(long tmsec = 0) const;
   34.83 +
   34.84 +	virtual void set_scaling(const Vector3 &pos, long tmsec = 0);
   34.85 +	virtual Vector3 get_node_scaling(long tmsec = 0) const;
   34.86 +
   34.87 +	// these take hierarchy into account
   34.88 +	virtual Vector3 get_position(long tmsec = 0) const;
   34.89 +	virtual Quaternion get_rotation(long tmsec = 0) const;
   34.90 +	virtual Vector3 get_scaling(long tmsec = 0) const;
   34.91 +
   34.92 +	virtual void set_pivot(const Vector3 &pivot);
   34.93 +	virtual Vector3 get_pivot() const;
   34.94 +
   34.95 +	// the local matrix is concatenated with the regular node/anim matrix
   34.96 +	virtual void set_local_matrix(const Matrix4x4 &mat);
   34.97 +	virtual const Matrix4x4 &get_local_matrix() const;
   34.98 +
   34.99 +	// for bone nodes, the transformation of the bone in bind position
  34.100 +	virtual void set_bone_matrix(const Matrix4x4 &bmat);
  34.101 +	virtual const Matrix4x4 &get_bone_matrix() const;
  34.102 +
  34.103 +	// node transformation alone
  34.104 +	virtual void get_node_xform(long tmsec, Matrix4x4 *mat, Matrix4x4 *inv_mat = 0) const;
  34.105 +
  34.106 +	// node transformation taking hierarchy into account
  34.107 +	virtual void get_xform(long tmsec, Matrix4x4 *mat, Matrix4x4 *inv_mat = 0) const;
  34.108 +};
  34.109 +
  34.110 +
  34.111 +class Track {
  34.112 +private:
  34.113 +	struct anm_track *trk;
  34.114 +	Interp interp;
  34.115 +	Extrap extrap;
  34.116 +
  34.117 +public:
  34.118 +	Track();
  34.119 +	~Track();
  34.120 +
  34.121 +	Track(const Track &trk);
  34.122 +	Track &operator =(const Track &trk);
  34.123 +
  34.124 +	void set_interpolator(Interp in);
  34.125 +	Interp get_interpolator() const;
  34.126 +	void set_extrapolator(Extrap ex);
  34.127 +	Extrap get_extrapolator() const;
  34.128 +
  34.129 +	void set_default(double def);
  34.130 +
  34.131 +	void set_value(float val, long tmsec = 0);
  34.132 +	float get_value(long tmsec = 0) const;
  34.133 +
  34.134 +	// the same as get_value
  34.135 +	float operator ()(long tmsec = 0) const;
  34.136 +};
  34.137 +
  34.138 +class Track3 {
  34.139 +private:
  34.140 +	Track track[3];
  34.141 +
  34.142 +public:
  34.143 +	void set_interpolator(Interp in);
  34.144 +	Interp get_interpolator() const;
  34.145 +	void set_extrapolator(Extrap ex);
  34.146 +	Extrap get_extrapolator() const;
  34.147 +
  34.148 +	void set_default(const Vector3 &def);
  34.149 +
  34.150 +	void set_value(const Vector3 &val, long tmsec = 0);
  34.151 +	Vector3 get_value(long tmsec = 0) const;
  34.152 +
  34.153 +	// the same as get_value
  34.154 +	Vector3 operator ()(long tmsec = 0) const;
  34.155 +};
  34.156 +
  34.157 +#endif	/* GOATGFX_XFORM_NODE_H_ */
    35.1 --- a/vrchess.sln	Sat Aug 23 12:03:29 2014 +0300
    35.2 +++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
    35.3 @@ -1,20 +0,0 @@
    35.4 -
    35.5 -Microsoft Visual Studio Solution File, Format Version 12.00
    35.6 -# Visual Studio 2012
    35.7 -Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "vrchess", "vrchess.vcxproj", "{714906B6-FD4C-4ECC-990C-247FA46B8801}"
    35.8 -EndProject
    35.9 -Global
   35.10 -	GlobalSection(SolutionConfigurationPlatforms) = preSolution
   35.11 -		Debug|Win32 = Debug|Win32
   35.12 -		Release|Win32 = Release|Win32
   35.13 -	EndGlobalSection
   35.14 -	GlobalSection(ProjectConfigurationPlatforms) = postSolution
   35.15 -		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Debug|Win32.ActiveCfg = Debug|Win32
   35.16 -		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Debug|Win32.Build.0 = Debug|Win32
   35.17 -		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Release|Win32.ActiveCfg = Release|Win32
   35.18 -		{714906B6-FD4C-4ECC-990C-247FA46B8801}.Release|Win32.Build.0 = Release|Win32
   35.19 -	EndGlobalSection
   35.20 -	GlobalSection(SolutionProperties) = preSolution
   35.21 -		HideSolutionNode = FALSE
   35.22 -	EndGlobalSection
   35.23 -EndGlobal
    36.1 --- a/vrchess.vcxproj	Sat Aug 23 12:03:29 2014 +0300
    36.2 +++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
    36.3 @@ -1,117 +0,0 @@
    36.4 -<?xml version="1.0" encoding="utf-8"?>
    36.5 -<Project DefaultTargets="Build" ToolsVersion="12.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
    36.6 -  <ItemGroup Label="ProjectConfigurations">
    36.7 -    <ProjectConfiguration Include="Debug|Win32">
    36.8 -      <Configuration>Debug</Configuration>
    36.9 -      <Platform>Win32</Platform>
   36.10 -    </ProjectConfiguration>
   36.11 -    <ProjectConfiguration Include="Release|Win32">
   36.12 -      <Configuration>Release</Configuration>
   36.13 -      <Platform>Win32</Platform>
   36.14 -    </ProjectConfiguration>
   36.15 -  </ItemGroup>
   36.16 -  <PropertyGroup Label="Globals">
   36.17 -    <ProjectGuid>{714906B6-FD4C-4ECC-990C-247FA46B8801}</ProjectGuid>
   36.18 -    <Keyword>Win32Proj</Keyword>
   36.19 -    <RootNamespace>vrchess</RootNamespace>
   36.20 -  </PropertyGroup>
   36.21 -  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
   36.22 -  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
   36.23 -    <ConfigurationType>Application</ConfigurationType>
   36.24 -    <UseDebugLibraries>true</UseDebugLibraries>
   36.25 -    <PlatformToolset>v120</PlatformToolset>
   36.26 -    <CharacterSet>MultiByte</CharacterSet>
   36.27 -  </PropertyGroup>
   36.28 -  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
   36.29 -    <ConfigurationType>Application</ConfigurationType>
   36.30 -    <UseDebugLibraries>false</UseDebugLibraries>
   36.31 -    <PlatformToolset>v120</PlatformToolset>
   36.32 -    <WholeProgramOptimization>false</WholeProgramOptimization>
   36.33 -    <CharacterSet>MultiByte</CharacterSet>
   36.34 -  </PropertyGroup>
   36.35 -  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
   36.36 -  <ImportGroup Label="ExtensionSettings">
   36.37 -  </ImportGroup>
   36.38 -  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
   36.39 -    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
   36.40 -  </ImportGroup>
   36.41 -  <ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
   36.42 -    <Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
   36.43 -  </ImportGroup>
   36.44 -  <PropertyGroup Label="UserMacros" />
   36.45 -  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
   36.46 -    <LinkIncremental>true</LinkIncremental>
   36.47 -  </PropertyGroup>
   36.48 -  <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
   36.49 -    <LinkIncremental>false</LinkIncremental>
   36.50 -  </PropertyGroup>
   36.51 -  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
   36.52 -    <ClCompile>
   36.53 -      <PrecompiledHeader>
   36.54 -      </PrecompiledHeader>
   36.55 -      <WarningLevel>Level3</WarningLevel>
   36.56 -      <Optimization>Disabled</Optimization>
   36.57 -      <PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions);USE_LIBOVR</PreprocessorDefinitions>
   36.58 -      <DisableSpecificWarnings>4996;4244;4305</DisableSpecificWarnings>
   36.59 -      <AdditionalIncludeDirectories>$(SolutionDir)\src</AdditionalIncludeDirectories>
   36.60 -    </ClCompile>
   36.61 -    <Link>
   36.62 -      <SubSystem>Console</SubSystem>
   36.63 -      <GenerateDebugInformation>true</GenerateDebugInformation>
   36.64 -      <AdditionalDependencies>opengl32.lib;freeglutd.lib;glew32.lib;libvmath.lib;libimago2.lib;jpeglib.lib;libpng.lib;zlib.lib;libovrd.lib;ws2_32.lib;%(AdditionalDependencies)</AdditionalDependencies>
   36.65 -    </Link>
   36.66 -  </ItemDefinitionGroup>
   36.67 -  <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
   36.68 -    <ClCompile>
   36.69 -      <WarningLevel>Level3</WarningLevel>
   36.70 -      <PrecompiledHeader>
   36.71 -      </PrecompiledHeader>
   36.72 -      <Optimization>MaxSpeed</Optimization>
   36.73 -      <FunctionLevelLinking>true</FunctionLevelLinking>
   36.74 -      <IntrinsicFunctions>true</IntrinsicFunctions>
   36.75 -      <PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions);USE_LIBOVR</PreprocessorDefinitions>
   36.76 -      <DisableSpecificWarnings>4996;4244;4305</DisableSpecificWarnings>
   36.77 -      <AdditionalIncludeDirectories>$(SolutionDir)\src</AdditionalIncludeDirectories>
   36.78 -    </ClCompile>
   36.79 -    <Link>
   36.80 -      <SubSystem>Console</SubSystem>
   36.81 -      <GenerateDebugInformation>true</GenerateDebugInformation>
   36.82 -      <EnableCOMDATFolding>true</EnableCOMDATFolding>
   36.83 -      <OptimizeReferences>true</OptimizeReferences>
   36.84 -      <AdditionalDependencies>opengl32.lib;freeglut.lib;glew32.lib;libvmath.lib;libimago2.lib;jpeglib.lib;libpng.lib;zlib.lib;libovr.lib;ws2_32.lib;%(AdditionalDependencies)</AdditionalDependencies>
   36.85 -    </Link>
   36.86 -  </ItemDefinitionGroup>
   36.87 -  <ItemGroup>
   36.88 -    <ClCompile Include="src\camera.cc" />
   36.89 -    <ClCompile Include="src\game.cc" />
   36.90 -    <ClCompile Include="src\image.cc" />
   36.91 -    <ClCompile Include="src\main.cc" />
   36.92 -    <ClCompile Include="src\opengl.cc" />
   36.93 -    <ClCompile Include="src\sdr.c" />
   36.94 -    <ClCompile Include="src\texture.cc" />
   36.95 -    <ClCompile Include="src\vr\mathutil.c" />
   36.96 -    <ClCompile Include="src\vr\opt.c" />
   36.97 -    <ClCompile Include="src\vr\rbtree.c" />
   36.98 -    <ClCompile Include="src\vr\vr.c" />
   36.99 -    <ClCompile Include="src\vr\vr_libovr.c" />
  36.100 -    <ClCompile Include="src\vr\vr_modules.c" />
  36.101 -    <ClCompile Include="src\vr\vr_null.c" />
  36.102 -    <ClCompile Include="src\vr\vr_openhmd.c" />
  36.103 -  </ItemGroup>
  36.104 -  <ItemGroup>
  36.105 -    <ClInclude Include="src\camera.h" />
  36.106 -    <ClInclude Include="src\game.h" />
  36.107 -    <ClInclude Include="src\image.h" />
  36.108 -    <ClInclude Include="src\opengl.h" />
  36.109 -    <ClInclude Include="src\sdr.h" />
  36.110 -    <ClInclude Include="src\texture.h" />
  36.111 -    <ClInclude Include="src\vr\mathutil.h" />
  36.112 -    <ClInclude Include="src\vr\opt.h" />
  36.113 -    <ClInclude Include="src\vr\rbtree.h" />
  36.114 -    <ClInclude Include="src\vr\vr.h" />
  36.115 -    <ClInclude Include="src\vr\vr_impl.h" />
  36.116 -  </ItemGroup>
  36.117 -  <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
  36.118 -  <ImportGroup Label="ExtensionTargets">
  36.119 -  </ImportGroup>
  36.120 -</Project>
  36.121 \ No newline at end of file
    37.1 --- a/vrchess.vcxproj.filters	Sat Aug 23 12:03:29 2014 +0300
    37.2 +++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
    37.3 @@ -1,94 +0,0 @@
    37.4 -<?xml version="1.0" encoding="utf-8"?>
    37.5 -<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
    37.6 -  <ItemGroup>
    37.7 -    <Filter Include="src">
    37.8 -      <UniqueIdentifier>{4FC737F1-C7A5-4376-A066-2A32D752A2FF}</UniqueIdentifier>
    37.9 -      <Extensions>cpp;c;cc;cxx;def;odl;idl;hpj;bat;asm;asmx;h;inl</Extensions>
   37.10 -    </Filter>
   37.11 -    <Filter Include="src\vr">
   37.12 -      <UniqueIdentifier>{e881ab02-1a45-43f6-a15d-ee7f77256a1e}</UniqueIdentifier>
   37.13 -    </Filter>
   37.14 -  </ItemGroup>
   37.15 -  <ItemGroup>
   37.16 -    <ClCompile Include="src\main.cc">
   37.17 -      <Filter>src</Filter>
   37.18 -    </ClCompile>
   37.19 -    <ClCompile Include="src\camera.cc">
   37.20 -      <Filter>src</Filter>
   37.21 -    </ClCompile>
   37.22 -    <ClCompile Include="src\opengl.cc">
   37.23 -      <Filter>src</Filter>
   37.24 -    </ClCompile>
   37.25 -    <ClCompile Include="src\sdr.c">
   37.26 -      <Filter>src</Filter>
   37.27 -    </ClCompile>
   37.28 -    <ClCompile Include="src\game.cc">
   37.29 -      <Filter>src</Filter>
   37.30 -    </ClCompile>
   37.31 -    <ClCompile Include="src\image.cc">
   37.32 -      <Filter>src</Filter>
   37.33 -    </ClCompile>
   37.34 -    <ClCompile Include="src\texture.cc">
   37.35 -      <Filter>src</Filter>
   37.36 -    </ClCompile>
   37.37 -    <ClCompile Include="src\vr\vr.c">
   37.38 -      <Filter>src\vr</Filter>
   37.39 -    </ClCompile>
   37.40 -    <ClCompile Include="src\vr\vr_libovr.c">
   37.41 -      <Filter>src\vr</Filter>
   37.42 -    </ClCompile>
   37.43 -    <ClCompile Include="src\vr\vr_modules.c">
   37.44 -      <Filter>src\vr</Filter>
   37.45 -    </ClCompile>
   37.46 -    <ClCompile Include="src\vr\vr_null.c">
   37.47 -      <Filter>src\vr</Filter>
   37.48 -    </ClCompile>
   37.49 -    <ClCompile Include="src\vr\rbtree.c">
   37.50 -      <Filter>src\vr</Filter>
   37.51 -    </ClCompile>
   37.52 -    <ClCompile Include="src\vr\opt.c">
   37.53 -      <Filter>src\vr</Filter>
   37.54 -    </ClCompile>
   37.55 -    <ClCompile Include="src\vr\vr_openhmd.c">
   37.56 -      <Filter>src\vr</Filter>
   37.57 -    </ClCompile>
   37.58 -    <ClCompile Include="src\vr\mathutil.c">
   37.59 -      <Filter>src\vr</Filter>
   37.60 -    </ClCompile>
   37.61 -  </ItemGroup>
   37.62 -  <ItemGroup>
   37.63 -    <ClInclude Include="src\camera.h">
   37.64 -      <Filter>src</Filter>
   37.65 -    </ClInclude>
   37.66 -    <ClInclude Include="src\game.h">
   37.67 -      <Filter>src</Filter>
   37.68 -    </ClInclude>
   37.69 -    <ClInclude Include="src\image.h">
   37.70 -      <Filter>src</Filter>
   37.71 -    </ClInclude>
   37.72 -    <ClInclude Include="src\opengl.h">
   37.73 -      <Filter>src</Filter>
   37.74 -    </ClInclude>
   37.75 -    <ClInclude Include="src\sdr.h">
   37.76 -      <Filter>src</Filter>
   37.77 -    </ClInclude>
   37.78 -    <ClInclude Include="src\texture.h">
   37.79 -      <Filter>src</Filter>
   37.80 -    </ClInclude>
   37.81 -    <ClInclude Include="src\vr\vr.h">
   37.82 -      <Filter>src\vr</Filter>
   37.83 -    </ClInclude>
   37.84 -    <ClInclude Include="src\vr\vr_impl.h">
   37.85 -      <Filter>src\vr</Filter>
   37.86 -    </ClInclude>
   37.87 -    <ClInclude Include="src\vr\rbtree.h">
   37.88 -      <Filter>src\vr</Filter>
   37.89 -    </ClInclude>
   37.90 -    <ClInclude Include="src\vr\opt.h">
   37.91 -      <Filter>src\vr</Filter>
   37.92 -    </ClInclude>
   37.93 -    <ClInclude Include="src\vr\mathutil.h">
   37.94 -      <Filter>src\vr</Filter>
   37.95 -    </ClInclude>
   37.96 -  </ItemGroup>
   37.97 -</Project>
   37.98 \ No newline at end of file