common/LittleOBJLoader.h
b23636c9
 #ifndef loadobj_h
 #define loadobj_h
 
 #ifdef __APPLE__
 	#include <OpenGL/gl3.h>
 #else
 	#if defined(_WIN32)
 		#include "glew.h"
 	#endif
 	#include <GL/gl.h>
 #endif
 
 #include "VectorUtils4.h"
 
 //#ifdef __cplusplus
 //extern "C" {
 //#endif
 
 // How many error messages do you want before it stops?
 #define NUM_DRAWMODEL_ERROR 8
 
 typedef struct Mtl
 {
 	char newmtl[255];
 	
 	vec3 Ka, Kd, Ks, Ke;
 	// I have also seen vec3's named Tf and a float called Ni in one file. What is that?
 	// Tf = transmission filter
 	// Ni = optical_density
 	GLfloat Ns, Tr, d; // Tr = 1-d
 	int illum;	// illumination model 0..10
 	char map_Ka[255], map_Kd[255], map_Ks[255], map_Ke[255], map_Ns[255], map_d[255], map_bump[255];
 	
 	// NEW for texture support
 	int texidKa, texidKd, texidKs, texidKe, texidNs, texid_d, texid_bump; // References to loaded textures!
 	// refl -type not supported -o for options? A whole bunch.
 	// Extensions for physically based rendering not supported
 } Mtl, *MtlPtr, **MtlHandle;
 
 typedef struct
 {
   vec3* vertexArray;
   vec3* normalArray;
   vec2* texCoordArray;
   vec3* colorArray; // Rarely used
   GLuint* indexArray;
   int numVertices;
   int numIndices;
   char externalData;
   
   // Space for saving VBO and VAO IDs
   GLuint vao; // VAO
   GLuint vb, ib, nb, tb; // VBOs
   
   Mtl *material;
 } Model;
 
 // Basic model loading
 #define LoadModelPlus LoadModel
 Model* LoadModel(const char* name); // Load OBJ as single Model
 Model** LoadModelSet(const char* name);  // Multi-part OBJ!
 
 // Drawing models
 void DrawModel(Model *m, GLuint program, const char* vertexVariableName, const char* normalVariableName, const char* texCoordVariableName);
 void DrawWireframeModel(Model *m, GLuint program, const char* vertexVariableName, const char* normalVariableName, const char* texCoordVariableName);
 
 // Utility functions that you may need if you want to modify the model.
 
 Model* LoadDataToModel(
 			vec3 *vertices,
 			vec3 *normals,
 			vec2 *texCoords,
 			vec3 *colors,
 			GLuint *indices,
 			int numVert,
 			int numInd);
 void ReloadModelData(Model *m);
 
 void CenterModel(Model *m);
 void ScaleModel(Model *m, float sx, float sy, float sz);
 void DisposeModel(Model *m);
 
 //#ifdef __cplusplus
 //}
 ///#endif
 
49ca15c4
 // --------------- Implementation part ----------------
 
 #ifdef LOL_IMPLEMENTATION
 #define LOL_IMPLEMENTATION
b23636c9
 
 // Little OBJ loader
 // Ingemar's little OBJ loader
 // Formerly named LoadOBJ or loadobj
 // by Ingemar Ragnemalm 2005, 2008, 2019, 2021
 
 // Original version 2005 was extremely simple and intended for use with old OpenGL immediate mode.
 // Additions and reorganization by Mikael Kalms 2008
 // 120913: Revised LoadModelPlus/DrawModel by Jens.
 // Partially corrected formatting. (It is a mess!)
 // 130227: Error reporting in DrawModel
 // 130422: Added ScaleModel
 // 150909: Frees up temporary "Mesh" memory i LoadModel. Thanks to Simon Keisala for finding this!
 // Added DisposeModel. Limited the number of error printouts, thanks to Rasmus Hytter for this suggestion!
 // 160302: Uses fopen_s on Windows, as suggested by Jesper Post. Should reduce warnings a bit.
 // 160510: Uses calloc instead of malloc (for safety) in many places where it could possibly cause problems.
 // 170406: Added "const" to string arguments to make C++ happier.
 // 190416: free mesh->coordStarts properly.
 // 191110: Finally! Multi-part OBJ files work! Renamed LoadModelPlus to LoadModel.
 // and added LoadModelSet! The old LoadModel was removed. Converted to use vec2/vec3
 // instead of arrays of float. It is now dependent on VectorUtils3. If you want to
 // make it independent of VectorUtils3, only a few structs and functions have to be replaced.
 // MTL support working! Parser totally rewritten. Officially renamed to Ingemars Little OBJ Loader,
 // LOL for short! And boy did I find a nice acronym!
 // 210125: Corrected types in LoadDataToModel. Corrected material disposal in the still badly tested DisposeModel.
 // 210218: Allows TAB between parts of string.
 // 210304: Corrected bad allocation size on row 538, found by Jens Lindgren, who also found glDeleteVertexArrays in DisposeModel.
 // 210422: Change by Felicia Castenbrandt, around row 415, copy path or source. This change is likely to be needed in some other places.
 // 210502: Dispose the material name list, omission found by Jens Lindgren.
 // 220222: Added a printed error message if ParseOBJ can't open a file. Also makes DrawModel always activare the shader for you.
 // 2022 somewhere: Made experimental header only version, necessary to work with VectorUtils in C++.
49ca15c4
 // 2023-02-05: Added LOL_IMPLEMENTATION to avoid double compilation.
b23636c9
 
 // Usage:
 // Load simple models with LoadModel. Multi-part models are loaded with LoadModelSet.
 // DrawModel will draw the model, or, for a multi-part model, one part.
 // If you need to modify a model on the CPU, you can do that and re-upload
 // with LoadDataToModel.
 // There is no longer any call that only loads the data, but you can
 // make your own by taking out the GPU upload from LoadModel.
 
 // This is free software. I appreciate credits in derivate products or
 // if it is used in public products.
 
 //#include "LittleOBJLoader.h"
 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
 #include <math.h>
 #include "VectorUtils4.h"
 
 #if defined(_WIN32)
 	#define sscanf sscanf_s
 #endif
 
 
 static char atLineEnd = 0;
 static char atFileEnd = 0;
 
 static char ParseString(char *line, int *pos, char *s)
 {
 	int i = 0;
 	char c;
 	c = line[(*pos)++];
 	while (c == 32 || c == 9) // Important change 210218
 		c = line[(*pos)++]; // Skip leading spaces
 	while (c != 0 && c != 32 && c != 9  && c != EOF && c != '/' && i < 254)
 	{
 		s[i++] = c;
 		c = line[(*pos)++];
 	}
 	atLineEnd = (c == 0);
 	s[i] = 0;
 	
 	return c;
 }
 
 static float ParseFloat(char *line, int *pos)
 {
 	float floatValue = 0.0;
 	char s[255];
 	ParseString(line, pos, s);
 	sscanf(s, "%f", &floatValue);
 	if (isnan(floatValue) || isinf(floatValue))
 		floatValue = 0.0;
 	return floatValue;
 }
 
 static int ParseInt(char *line, int *pos, char *endc)
 {
 	int intValue = 0;
 	char s[255];
 	char c = ParseString(line, pos, s);
 	if (endc != NULL) *endc = c;
 	if (strlen(s) == 0)
 		intValue = -1; // Nothing found!
 	else
 		sscanf(s, "%d", &intValue);
 	return intValue;
 }
 
 static vec3 ParseVec3(char *line, int *pos)
 {
 	vec3 v;
 	v.x = ParseFloat(line, pos);
 	v.y = ParseFloat(line, pos);
 	v.z = ParseFloat(line, pos);
 	return v;
 }
 
 void ParseLine(FILE *fp, char *line)
 {
 	int i, j;
 	char c = '_';
 	for (i = 0, j= 0; i < 2048; i++)
 	{
 		c = getc(fp);
 		if (c != 10 && c != 13)
 			line[j++] = c;
 		if (c == EOF || ((c == 10 || c == 13) && j > 0))
 		{
 			if (c == EOF)
 				atFileEnd = 1;
 			line[j] = 0;
 			return;
 		}
 	}
 	line[j] = 0;
 	return;
 }
 
 
 // Dispose the strings in the materials list
 static void DisposeMtlList(Mtl **materials)
 {
 	if (materials != NULL)
 	for (int i = 0; materials[i] != NULL; i++)
 		free(materials[i]);
 	free(materials);
 }
 
 static Mtl **ParseMTL(char *filename)
 {
 	Mtl *m;
 	char s[255];
 	char line[2048];
 	int pos;
 	Mtl **materials = NULL;
 	FILE *fp;
 
 	// It seems Windows/VS doesn't like fopen any more, but fopen_s is not on the others.
 	#if defined(_WIN32)
 	fopen_s(&fp, filename, "r");
 	#else
 	fp = fopen(filename, "rb"); // rw works everywhere except Windows?
 	#endif
 	if (fp == NULL)
 		return NULL;
 	atLineEnd = 0;
 	atFileEnd = 0;
 	int matcount = 0;
 
 	while (!atFileEnd)
 	{
 		ParseLine(fp, line);
 		pos = 0;
 		ParseString(line, &pos, s);
 		if (strcmp(s, "newmtl") == 0)
 			matcount++;
 	}
 	rewind(fp);
 
 	materials = (Mtl **)calloc(sizeof(MtlPtr)*(matcount+1), 1);
 
 	matcount = 0;
 	atLineEnd = 0;
 	atFileEnd = 0;
 	
 	while (!atFileEnd)
 	{
 		ParseLine(fp, line);
 		pos = 0;
 		ParseString(line, &pos, s);
 
 		// NEW MATERIAL!
 		if (strcmp(s, "newmtl") == 0)
 		{
 			matcount++;
 			materials[matcount-1] = (Mtl *)calloc(sizeof(Mtl),1);
 			m = materials[matcount-1];
 			materials[matcount] = NULL;
 			
 			ParseString(line, &pos, m->newmtl);
 		}
 
 		if (m != NULL)
 		{
 			if (strcmp(s, "Ka") == 0)
 				m->Ka = ParseVec3(line, &pos);
 			if (strcmp(s, "Kd") == 0)
 				m->Kd = ParseVec3(line, &pos);
 			if (strcmp(s, "Ks") == 0)
 				m->Ks = ParseVec3(line, &pos);
 			if (strcmp(s, "Ke") == 0)
 				m->Ke = ParseVec3(line, &pos);
 			if (strcmp(s, "Tr") == 0)
 			{
 				m->Tr = ParseFloat(line, &pos);
 				m->d = 1 - m->Tr;
 			}
 			if (strcmp(s, "d") == 0)
 			{
 				m->d = ParseFloat(line, &pos);
 				m->Tr = 1 - m->d;
 			}
 
 			if (strcmp(s, "illum") == 0)
 				m->illum = ParseInt(line, &pos, NULL);
 
 			if (strcmp(s, "map_Ka") == 0)
 				ParseString(line, &pos, m->map_Ka);
 			if (strcmp(s, "map_Kd") == 0)
 				ParseString(line, &pos, m->map_Kd);
 			if (strcmp(s, "map_Ks") == 0)
 				ParseString(line, &pos, m->map_Ks);
 			if (strcmp(s, "map_Ke") == 0)
 				ParseString(line, &pos, m->map_Ke);
 			if (strcmp(s, "map_d") == 0)
 				ParseString(line, &pos, m->map_d);
 			if (strcmp(s, "map_bump") == 0)
 				ParseString(line, &pos, m->map_bump);
 			if (strcmp(s, "bump") == 0)
 				ParseString(line, &pos, m->map_bump);
 		}
 	}
 	fclose(fp);
 
 	return materials;
 }
 
 typedef struct Mesh
 {
 	vec3	*vertices;
 	int		vertexCount;
 	vec3	*vertexNormals;
 	int		normalsCount;
 	vec2	*textureCoords;
 	int		texCount;
 	
 	int		*coordIndex;
 	int		*normalsIndex;
 	int		*textureIndex;
 	int		coordCount; // Number of indices in each index struct
 	
 	// Borders between groups
 	int		*coordStarts;
 	int		groupCount;
 	
 	char	*materialName;
 } Mesh, *MeshPtr;
 
 // Globals set by LoadObj
 static int vertCount, texCount, normalsCount, coordCount;
 static int zeroFix;
 static char hasPositionIndices;
 static char hasNormalIndices;
 static char hasTexCoordIndices;
 
 // List of materials
 Mtl **gMaterials;
 char **gMtlNameList;
 char *gMtlLibName = NULL;
 
 static char ParseOBJ(const char *filename, MeshPtr theMesh)
 {
 	int lastCoordCount = -1;
 	FILE *fp;
 	
 	// It seems Windows/VS doesn't like fopen any more, but fopen_s is not on the others.
 	#if defined(_WIN32)
 	fopen_s(&fp, filename, "r");
 	#else
 	fp = fopen(filename, "rb"); // rw works everywhere except Windows?
 	#endif
 	if (fp == NULL)
 	{
 		fprintf(stderr, "File \"%s\" could not be opened\n", filename);
 		return -1;
 	}
 
 // These must be taken care of by the caller!
 	vertCount=0;
 	texCount=0;
 	normalsCount=0;
 	coordCount=0;
 
 	atLineEnd = 0;
 	atFileEnd = 0;
 		
 	while (!atFileEnd)
 	{
 		char line[2048];
 		ParseLine(fp, line);
 		int pos = 0;
 		
 		char s[256];
 		ParseString(line, &pos, s);
 		
 		if (strcmp(s, "v") == 0)
 		{
 			vec3 v = ParseVec3(line, &pos);
 			if (theMesh->vertices != NULL)
 				theMesh->vertices[vertCount++] = v;
 			else
 				vertCount += 1;
 		}
 		if (strcmp(s, "vn") == 0)
 		{
 			vec3 v = ParseVec3(line, &pos);			
 			if (theMesh->vertexNormals != NULL)
 				theMesh->vertexNormals[normalsCount++] = v;
 			else
 				normalsCount += 1;
 		}
 		if (strcmp(s, "vt") == 0)
 		{
 			vec3 v = ParseVec3(line, &pos);
 			if (theMesh->textureCoords != NULL)
 			{
 				theMesh->textureCoords[texCount].x = v.x;
 				theMesh->textureCoords[texCount++].y = v.y;
 			}
 			else
 				texCount += 1;
 		}
 		if (strcmp(s, "f") == 0)
 		{
 		// Unknown number of singeltons or triplets!
 			
 			int p = -1;
 			int i = -1;
 			int p1;
 			char *ns;
 			char done = 0;
 			for (i = pos ; !done; pos++)
 			{
 				if (line[pos] == 0)
 					done = 1;
 				if (p >= 0) // in triplet)
 				{
 					char c = line[pos];
 					if (c == '/' || c == ' ' || c == 0)
 					{
 						ns = &line[p1]; // Start of substring
 						line[pos] = 0; // End of substring
 						if (p1 != pos)
 						{
 							i = atoi(ns);
 							if (i == 0)
 								zeroFix = 1;
 							if (i < 0)
 							{
 								switch (p)
 								{
 									case 0: i = vertCount + i + 1; break;
 									case 1: i = texCount + i + 1; break;
 									case 2: i = normalsCount + i + 1; break;
 								}
 							}
 							i = i + zeroFix;
 						}
 						else
 							i = 0; // -1
 						
 						if (i > 0)
 						switch (p)
 						{
 							case 0:
 								if (theMesh->coordIndex != NULL)
 									theMesh->coordIndex[coordCount-1] = i-1;
 								break;
 							case 1:
 								if (theMesh->textureIndex != NULL)
 									if (i >= 0)
 										theMesh->textureIndex[coordCount-1] = i-1;
 								hasTexCoordIndices = 1;
 								break;
 							case 2:
 								if (theMesh->normalsIndex != NULL)
 									if (i >= 0)
 										theMesh->normalsIndex[coordCount-1] = i-1;
 								hasNormalIndices = 1;
 								break;
 						}
 						p1 = pos + 1;
 						p++;
 					}
 					if (c == ' ')
 					{
 						p = -1; // Next triplet
 					}
 				}
 				else // Wait for non-space
 				if (line[pos] != ' ')
 				{
 					if (!done)
 						coordCount++;
 					p = 0; // Triplet starts!
 					p1 = pos;
 				}
 			}
 			
 			if (theMesh->coordIndex != NULL)
 				theMesh->coordIndex[coordCount] = -1;
 			if (theMesh->textureIndex != NULL)
 				theMesh->textureIndex[coordCount] = -1;
 			if (theMesh->normalsIndex != NULL)
 				theMesh->normalsIndex[coordCount] = -1;
 			coordCount++;
 		}
 		if (strcmp(s, "mtllib") == 0)
 		{
 			// Save file name for later
 			char libname[256];
 			
 			// Added by Felicia Castenbrandt: Get path of main file, if any
         	char temp[255];
 			int index = 0;
 
 			//find index of last /
 			for(unsigned int i = 0; i < strlen(filename); i++)
 				if(filename[i] == '/')
 					index = i;
 
 			//copy everything before the last / to temp
 			if (index != 0)
 			{
 				for(int i = 0; i < index+1; i++)
 				{
 					char ch = filename[i];
 					strncat(temp, &ch, 1);
 				}
 			}
 			// End of addition
 			
 			
 			/*char c =*/ ParseString(line, &pos, libname);
 			gMtlLibName = (char *)malloc(strlen(libname)+1);
 #if defined(_WIN32)
 			strcpy_s(gMtlLibName, strlen(libname) + 1, libname);
 #else
 			strcpy(gMtlLibName, libname);
 #endif
 
 		}
 		if (strcmp(s, "usemtl") == 0)
 		{
 			// Groups are only identified by usemtl!
 			
 			if (coordCount > 0) // If no data has been seen, this must be the first group!
 			{
 				if (lastCoordCount != coordCount) // must not be the same; empty group (happens if a model has both "g" and "o")
 				{
 					theMesh->groupCount += 1;
 					if (theMesh->coordStarts != NULL) // NULL if we are just counting
 						theMesh->coordStarts[theMesh->groupCount] = coordCount;
 
 					lastCoordCount = coordCount;
 				}
 				else
 					printf("Ignored part!\n");
 			}
 			
 			if (gMtlNameList != NULL)
 			{
 				// Save matname for later
 				char matname[255];
 				ParseString(line, &pos, matname);
 				gMtlNameList[theMesh->groupCount] = (char *)malloc(strlen(matname)+1);
 #if defined(_WIN32)
 				strcpy_s(gMtlNameList[theMesh->groupCount], strlen(matname) + 1, matname);
 #else
 				strcpy(gMtlNameList[theMesh->groupCount], matname);
 #endif
 
 
 			}
 		}
 	} // While
 	
 	// Add last groups *if* there was more data since last time!
 	if (coordCount > lastCoordCount)
 	{
 		theMesh->groupCount += 1; // End of last group
 		if (theMesh->coordStarts != NULL)
 			theMesh->coordStarts[theMesh->groupCount] = coordCount;
 	}
 		
 	fclose(fp);
 	
 	if (gMtlLibName != NULL)
 		gMaterials = ParseMTL(gMtlLibName);
 	// else try based on filename: filename.obj -> filename.mtl or filename_obj.mtl
 	if (gMaterials == NULL)
 	if (strlen(filename) > 4)
 	{
 		char tryname[255];
 		strcpy(tryname, filename);
 		tryname[strlen(tryname) - 4] = '_';
 		strcat(tryname, ".mtl");
 		gMaterials = ParseMTL(tryname);
 	}
 	if (gMaterials == NULL)
 	if (strlen(filename) > 4)
 	{
 		char tmpname[255];
 		strcpy(tmpname, filename);
 		tmpname[strlen(tmpname) - 4] = 0;
 		strcat(tmpname, ".mtl");
 		gMaterials = ParseMTL(tmpname);
 	}
 	return 0;
 }
 
 // Load raw, unprocessed OBJ data!
 static struct Mesh * LoadOBJ(const char *filename)
 {
 	// Reads once to find sizes, again to fill buffers
 	Mesh *theMesh;
 	
 	// Allocate Mesh but not the buffers
 	theMesh = (Mesh *)calloc(sizeof(Mesh), 1);
 
 	hasPositionIndices = 1;
 	hasTexCoordIndices = 0;
 	hasNormalIndices = 0;
 
 	theMesh->coordStarts = NULL;
 	theMesh->groupCount = 0;
 	gMtlNameList = NULL;
 
 // Dispose if they exist!
 	gMtlLibName = NULL;
 	gMaterials = NULL;
 
 	vertCount=0;
 	texCount=0;
 	normalsCount=0;
 	coordCount=0;
 	zeroFix = 0;
 		
 	// Parse for size
 	ParseOBJ(filename, theMesh);
 
 	// Allocate arrays!
 	if (vertCount > 0)
 		theMesh->vertices = (vec3 *)malloc(sizeof(vec3) * vertCount);
 	if (texCount > 0)
 		theMesh->textureCoords = (vec2 *)malloc(sizeof(vec2) * texCount);
 	if (normalsCount > 0)
 		theMesh->vertexNormals = (vec3 *)malloc(sizeof(vec3) * normalsCount);
 	if (hasPositionIndices)
 		theMesh->coordIndex = (int *)calloc(coordCount, sizeof(int));
 	if (hasNormalIndices)
 		theMesh->normalsIndex = (int *)calloc(coordCount, sizeof(int));
 	if (hasTexCoordIndices)
 		theMesh->textureIndex = (int *)calloc(coordCount, sizeof(int));
 
 	gMtlNameList = (char **)calloc(sizeof(char *) * (theMesh->groupCount+1), 1);
 	theMesh->coordStarts = (int *)calloc(sizeof(int) * (theMesh->groupCount+2), 1); // Length found on last pass
 	theMesh->groupCount = 0;
 
 	// Zero again
 	vertCount=0;
 	texCount=0;
 	normalsCount=0;
 	coordCount=0;
 
 	// Parse again for filling buffers
 	ParseOBJ(filename, theMesh);
 	
 	theMesh->vertexCount = vertCount;
 	theMesh->coordCount = coordCount;
 	theMesh->texCount = texCount;
 	theMesh->normalsCount = normalsCount;
 
 	return theMesh;
 }
 
 void DecomposeToTriangles(struct Mesh *theMesh)
 {
 	int i, vertexCount, triangleCount;
 	int *newCoords, *newNormalsIndex, *newTextureIndex;
 	int newIndex = 0; // Index in newCoords
 	int first = 0;
 
 	// 1.1 Calculate how big the list will become
 	
 	vertexCount = 0; // Number of vertices in current polygon
 	triangleCount = 0; // Resulting number of triangles
 	for (i = 0; i < theMesh->coordCount; i++)
 	{
 		if (theMesh->coordIndex[i] == -1)
 		{
 		if (vertexCount > 2) triangleCount += vertexCount - 2;
 			vertexCount = 0;
 		}
 		else
 		{
 			vertexCount = vertexCount + 1;
 		}
 	}
 		
 	newCoords = (int *)calloc(triangleCount * 3, sizeof(int));
 	if (theMesh->normalsIndex != NULL)
 		newNormalsIndex = (int *)calloc(triangleCount * 3, sizeof(int));
 	if (theMesh->textureIndex != NULL)
 		newTextureIndex = (int *)calloc(triangleCount * 3, sizeof(int));
 	
 	// 1.2 Loop through old list and write the new one
 	// Almost same loop but now it has space to write the result
 	vertexCount = 0;
 	for (i = 0; i < theMesh->coordCount; i++)
 	{
 		if (theMesh->coordIndex[i] == -1)
 		{
 			first = i + 1;
 			vertexCount = 0;
 		}
 		else
 		{
 			vertexCount = vertexCount + 1;
 			if (vertexCount > 2)
 			{
 				newCoords[newIndex++] = theMesh->coordIndex[first];
 				newCoords[newIndex++] = theMesh->coordIndex[i-1];
 				newCoords[newIndex++] = theMesh->coordIndex[i];
 				if (theMesh->normalsIndex != NULL)
 				{
 					newNormalsIndex[newIndex-3] = theMesh->normalsIndex[first];
 					newNormalsIndex[newIndex-2] = theMesh->normalsIndex[i-1];
 					newNormalsIndex[newIndex-1] = theMesh->normalsIndex[i];
 				}
 				if (theMesh->textureIndex != NULL)
 				{
 					newTextureIndex[newIndex-3] = theMesh->textureIndex[first];
 					newTextureIndex[newIndex-2] = theMesh->textureIndex[i-1];
 					newTextureIndex[newIndex-1] = theMesh->textureIndex[i];
 				}
 			}
 		}
 	}
 	
 	free(theMesh->coordIndex);
 	theMesh->coordIndex = newCoords;
 	theMesh->coordCount = triangleCount * 3;
 	if (theMesh->normalsIndex != NULL)
 	{
 		free(theMesh->normalsIndex);
 		theMesh->normalsIndex = newNormalsIndex;
 	}
 	if (theMesh->textureIndex != NULL)
 	{
 		free(theMesh->textureIndex);
 		theMesh->textureIndex = newTextureIndex;
 	}
 } // DecomposeToTriangles
 
 static void GenerateNormals(Mesh* mesh)
 {
 	// If model has vertices but no vertexnormals, generate normals
 	if (mesh->vertices && !mesh->vertexNormals)
 	{
 		int face;
 		int normalIndex;
 
 		mesh->vertexNormals = (vec3 *)calloc(sizeof(vec3) * mesh->vertexCount, 1);
 		mesh->normalsCount = mesh->vertexCount;
 
 		mesh->normalsIndex = (int *)calloc(mesh->coordCount, sizeof(GLuint));
 		memcpy(mesh->normalsIndex, mesh->coordIndex, sizeof(GLuint) * mesh->coordCount);
 
 		for (face = 0; face * 3 < mesh->coordCount; face++)
 		{
 			int i0 = mesh->coordIndex[face * 3 + 0];
 			int i1 = mesh->coordIndex[face * 3 + 1];
 			int i2 = mesh->coordIndex[face * 3 + 2];
 			
 			vec3 v0 = VectorSub(mesh->vertices[i1], mesh->vertices[i0]);
 			vec3 v1 = VectorSub(mesh->vertices[i2], mesh->vertices[i0]);
 			vec3 v2 = VectorSub(mesh->vertices[i2], mesh->vertices[i1]);
 
 			float sqrLen0 = dot(v0, v0);
 			float sqrLen1 = dot(v1, v1);
 			float sqrLen2 = dot(v2, v2);
 
 			float len0 = (sqrLen0 >= 1e-6f) ? sqrt(sqrLen0) : 1e-3f;
 			float len1 = (sqrLen1 >= 1e-6f) ? sqrt(sqrLen1) : 1e-3f;
 			float len2 = (sqrLen2 >= 1e-6f) ? sqrt(sqrLen2) : 1e-3f;
 
 			float influence0 = dot(v0, v1) / (len0 * len1);
 			float influence1 = -dot(v0, v2) / (len0 * len2);
 			float influence2 = dot(v1, v2) / (len1 * len2);
 
 			float angle0 = (influence0 >= 1.f) ? 0 : 
 				(influence0 <= -1.f) ? M_PI : acos(influence0);
 			float angle1 = (influence1 >= 1.f) ? 0 : 
 				(influence1 <= -1.f) ? M_PI : acos(influence1);
 			float angle2 = (influence2 >= 1.f) ? 0 : 
 				(influence2 <= -1.f) ? M_PI : acos(influence2);
 
 			vec3 normal = cross(v0, v1);
 			mesh->vertexNormals[i0] = VectorAdd(ScalarMult(normal, angle0), mesh->vertexNormals[i0]);
 			mesh->vertexNormals[i1] = VectorAdd(ScalarMult(normal, angle1), mesh->vertexNormals[i1]);
 			mesh->vertexNormals[i2] = VectorAdd(ScalarMult(normal, angle2), mesh->vertexNormals[i2]);
 		}
 
 		for (normalIndex = 0; normalIndex < mesh->normalsCount; normalIndex++)
 		{
 			float length = Norm(mesh->vertexNormals[normalIndex]);
 			if (length > 0.01f)
 				mesh->vertexNormals[normalIndex] = ScalarMult(mesh->vertexNormals[normalIndex], 1/length);
 		}
 	}
 }
 
 static Model* GenerateModel(Mesh* mesh)
 {
 	// Convert from Mesh format (multiple index lists) to Model format
 	// (one index list) by generating a new set of vertices/indices
 	// and where new vertices have been created whenever necessary
 
 	typedef struct
 	{
 		int positionIndex;
 		int normalIndex;
 		int texCoordIndex;
 		int newIndex;
 	} IndexTriplet;
 
 	int hashGap = 6;
 
 	int indexHashMapSize = (mesh->vertexCount * hashGap + mesh->coordCount);
 
 	IndexTriplet* indexHashMap = (IndexTriplet *)malloc(sizeof(IndexTriplet)
 							* indexHashMapSize);
 
 	int numNewVertices = 0;
 	int index;
 
 	int maxValue = 0;
 		
 	Model* model = (Model *)malloc(sizeof(Model));
 	memset(model, 0, sizeof(Model));
 
 	model->indexArray = (GLuint *)malloc(sizeof(GLuint) * mesh->coordCount);
 	model->numIndices = mesh->coordCount;
 
 	memset(indexHashMap, 0xff, sizeof(IndexTriplet) * indexHashMapSize);
 
 	for (index = 0; index < mesh->coordCount; index++)
 	{
 		IndexTriplet currentVertex = { -1, -1, -1, -1 };
 		int insertPos = 0;
 		if (mesh->coordIndex)
 			currentVertex.positionIndex = mesh->coordIndex[index];
 		if (mesh->normalsIndex)
 			currentVertex.normalIndex = mesh->normalsIndex[index];
 		if (mesh->textureIndex)
 			currentVertex.texCoordIndex = mesh->textureIndex[index];
 
 		if (maxValue < currentVertex.texCoordIndex)
 			maxValue = currentVertex.texCoordIndex;
  
 		if (currentVertex.positionIndex >= 0)
 			insertPos = currentVertex.positionIndex * hashGap;
 
 		while (1)
 		{
 			if (indexHashMap[insertPos].newIndex == -1)
 				{
 					currentVertex.newIndex = numNewVertices++;
 					indexHashMap[insertPos] = currentVertex;
 					break;
 				}
 			else if (indexHashMap[insertPos].positionIndex
 				 == currentVertex.positionIndex
 				 && indexHashMap[insertPos].normalIndex
 				 == currentVertex.normalIndex
 				 && indexHashMap[insertPos].texCoordIndex
 				 == currentVertex.texCoordIndex)
 				{
 					currentVertex.newIndex = indexHashMap[insertPos].newIndex;
 					break;
 				}
 			else
 				insertPos++;
 		} 
 
 		model->indexArray[index] = currentVertex.newIndex;
 	}
 
 	if (mesh->vertices)
 		model->vertexArray = (vec3 *)malloc(sizeof(vec3) * numNewVertices);
 	if (mesh->vertexNormals)
 		model->normalArray = (vec3 *)malloc(sizeof(vec3) * numNewVertices);
 	if (mesh->textureCoords)
 		model->texCoordArray = (vec2 *)malloc(sizeof(vec2) * numNewVertices);
 	
 	model->numVertices = numNewVertices;
 
 	for (index = 0; index < indexHashMapSize; index++)
 	{
 		if (indexHashMap[index].newIndex != -1)
 		{
 			if (mesh->vertices)
 				model->vertexArray[indexHashMap[index].newIndex] = mesh->vertices[indexHashMap[index].positionIndex];
 			if (mesh->vertexNormals)
 			{
 				model->normalArray[indexHashMap[index].newIndex] = mesh->vertexNormals[indexHashMap[index].normalIndex];
 			}
 			if (mesh->textureCoords)
 				model->texCoordArray[indexHashMap[index].newIndex] = mesh->textureCoords[indexHashMap[index].texCoordIndex];
 		}
 	}
 
 	free(indexHashMap);
 
 	// If there is a material set, match materials to parts
 	if (gMaterials != NULL)
 	if (mesh->materialName != NULL)
 		for (int ii = 0; gMaterials[ii] != NULL; ii++)
 		{
 			Mtl *mtl = gMaterials[ii];
 			if (strcmp(mesh->materialName, mtl->newmtl) == 0)
 			{
 				// Copy mtl to model!
 				model->material = (Mtl *)malloc(sizeof(Mtl));
 				memcpy(model->material, mtl, sizeof(Mtl));
 
 				strcpy(model->material->map_Ka, mtl->map_Ka);
 				strcpy(model->material->map_Kd, mtl->map_Kd);
 				strcpy(model->material->map_Ks, mtl->map_Ks);
 				strcpy(model->material->map_Ke, mtl->map_Ke);
 				strcpy(model->material->map_Ns, mtl->map_Ns);
 				strcpy(model->material->map_d, mtl->map_d);
 				strcpy(model->material->map_bump, mtl->map_bump);
 			}
 		}
 
 	return model;
 }
 
 Mesh **SplitToMeshes(Mesh *m)
 {
 	int * mapc = (int *)malloc(m->vertexCount * sizeof(int));
 	int * mapt = (int *)malloc(m->texCount * sizeof(int));
 	int * mapn = (int *)malloc(m->normalsCount * sizeof(int));
 	
 	if (m == NULL || m ->vertices == NULL || m->vertexCount == 0)
 	{
 		printf("Illegal mesh!\n");
 		return NULL;
 	}
 	
 	Mesh **mm = (Mesh **)calloc(sizeof(Mesh *), m->groupCount+2);
 
 	for (int mi = 0; mi < m->groupCount; mi++) // For all sections
 	{
 		int from = m->coordStarts[mi];
 		int to = m->coordStarts[mi+1];
 		
 		mm[mi] = (Mesh *)calloc(sizeof(Mesh), 1); // allocate struct
 		
 		// Fill mapc, mapt, mapn with -1 (illegal index)
 		for (int ii = 0; ii < m->vertexCount; ii++)
 			mapc[ii] = -1;
 		for (int ii = 0; ii < m->texCount; ii++)
 			mapt[ii] = -1;
 		for (int ii = 0; ii < m->normalsCount; ii++)
 			mapn[ii] = -1;
 		
 		// Count number of entries needed
 		int intVertexCount = 0;
 		int intTexCount = 0;
 		int intNormalsCount = 0;
 		
 		for (int j = from; j < to; j++) // For all index triplets
 		{
 			int ix = m->coordIndex[j];
 			if (ix > -1)
 				if (mapc[ix] == -1)
 				{
 					mapc[ix] = ix;
 					intVertexCount++;
 				}
 			if (m->textureIndex != NULL)
 			{
 				ix = m->textureIndex[j];
 				if (ix > -1)
 					if (mapt[ix] == -1)
 					{
 						mapt[ix] = ix;
 						intTexCount++;
 					}
 			}
 			if (m->normalsIndex != NULL)
 			{
 				ix = m->normalsIndex[j];
 				if (ix > -1)
 					if (mapn[ix] == -1)
 					{
 						mapn[ix] = ix;
 						intNormalsCount++;
 					}
 			}
 		}
 				
 		// Allocate buffers
 		mm[mi]->coordIndex = (int *)malloc((to - from) * sizeof(int));
 		mm[mi]->textureIndex = (int *)malloc((to - from) * sizeof(int));
 		mm[mi]->normalsIndex = (int *)malloc((to - from) * sizeof(int));
 		if (intVertexCount > 0)
 			mm[mi]->vertices = (vec3 *)malloc(intVertexCount * sizeof(vec3));
 		if (intTexCount > 0)
 			mm[mi]->textureCoords = (vec2 *)malloc(intTexCount * sizeof(vec2));
 		if (intNormalsCount > 0)
 			mm[mi]->vertexNormals = (vec3 *)malloc(intNormalsCount * sizeof(vec3));
 		mm[mi]->vertexCount = intVertexCount;
 		mm[mi]->texCount = intTexCount;
 		mm[mi]->normalsCount = intNormalsCount;
 		
 		// Fill mapc, mapt, mapn with -1 (illegal index)
 		for (int ii = 0; ii < m->vertexCount; ii++)
 			mapc[ii] = -1;
 		for (int ii = 0; ii < m->texCount; ii++)
 			mapt[ii] = -1;
 		for (int ii = 0; ii < m->normalsCount; ii++)
 			mapn[ii] = -1;
 		
 		int mapcCount = 0;
 		int maptCount = 0;
 		int mapnCount = 0;
 		for (int j = from; j < to; j++) // For all index triplets
 		{
 			int ix = m->coordIndex[j];
 			if (ix > -1)
 			{
 				if (mapc[ix] == -1) // Unmapped
 				{
 					mapc[ix] = mapcCount++;
 					mm[mi]->vertices[mapc[ix]] = m->vertices[ix]; // Copy vertex to mm[i]
 				}
 				mm[mi]->coordIndex[j-from] = mapc[ix];
 			}
 			else // Separator
 			{
 				mm[mi]->coordIndex[j-from] = -1;
 			}
 
 			if (m->textureIndex != NULL)
 			if (mm[mi]->textureIndex != NULL)
 			{
 				ix = m->textureIndex[j];
 				if (ix > -1)
 				{
 					if (mapt[ix] == -1) // Unmapped
 					{
 						mapt[ix] = maptCount++;
 						mm[mi]->textureCoords[mapt[ix]] = m->textureCoords[ix]; // Copy vertex to mm[i]
 					}
 					mm[mi]->textureIndex[j-from] = mapt[ix];
 				}
 				else // Separator
 				{
 					mm[mi]->textureIndex[j-from] = -1;
 				}
 			}
 
 			if (m->normalsIndex != NULL)
 			if (mm[mi]->normalsIndex != NULL)
 			{
 				ix = m->normalsIndex[j];
 				if (ix > -1)
 				{
 					if (mapn[ix] == -1) // Unmapped
 					{
 						mapn[ix] = mapnCount++;
 						mm[mi]->vertexNormals[mapn[ix]] = m->vertexNormals[ix]; // Copy vertex to mm[i]
 					}
 					mm[mi]->normalsIndex[j-from] = mapn[ix];
 				}
 				else // Separator
 				{
 					mm[mi]->normalsIndex[j-from] = -1;
 				}
 			}
 		} // for all index triplets
 		mm[mi]->coordCount = to - from;
 		
 		if (gMtlNameList != NULL)
 		{
 			mm[mi]->materialName = gMtlNameList[mi];
 			gMtlNameList[mi] = NULL; // No longer "owned" by the gMtlNameList
 		}
 	} // for all parts
 	
 	return mm;
 }
 
 
 static void DisposeMesh(Mesh *mesh)
 {
 // Free the mesh!
 	if (mesh != NULL)
 	{
 		if (mesh->vertices != NULL)
 			free(mesh->vertices);
 		if (mesh->vertexNormals != NULL)
 			free(mesh->vertexNormals);
 		if (mesh->textureCoords != NULL)
 			free(mesh->textureCoords);
 		if (mesh->coordIndex != NULL)
 			free(mesh->coordIndex);
 		if (mesh->normalsIndex != NULL)
 			free(mesh->normalsIndex);
 		if (mesh->textureIndex != NULL)
 			free(mesh->textureIndex);
 #if !defined(_WIN32)
 // This is very disturbing, causes heap corruption on Windows. Reason not found.
 		if (mesh->coordStarts != NULL)
 			free(mesh->coordStarts);
 #endif
 		if (mesh->materialName != NULL)
 			free(mesh->materialName);
 
 // Dispose the material name list too
 		if (gMtlNameList != NULL)
 		{
 			for (int i = 0; i < mesh->groupCount; i++)
 				if (gMtlNameList[i] != NULL)
 					free(gMtlNameList[i]);
 			free(gMtlNameList);
 			gMtlNameList = NULL;
 		}
 
 		free(mesh);
 	}
 }
 
 void CenterModel(Model *m)
 {
 	int i;
 	float maxx = -1e10, maxy = -1e10, maxz = -1e10, minx = 1e10, miny = 1e10, minz = 1e10;
 	
 	for (i = 0; i < m->numVertices; i++)
 	{
 		if (m->vertexArray[i].x < minx) minx = m->vertexArray[i].x;
 		if (m->vertexArray[i].x > maxx) maxx = m->vertexArray[i].x;
 		if (m->vertexArray[i].y < miny) miny = m->vertexArray[i].y;
 		if (m->vertexArray[i].y > maxy) maxy = m->vertexArray[i].y;
 		if (m->vertexArray[i].z < minz) minz = m->vertexArray[i].z;
 		if (m->vertexArray[i].z > maxz) maxz = m->vertexArray[i].z;
 	}
 	
 	for (i = 0; i < m->numVertices; i++)
 	{
 		m->vertexArray[i].x -= (maxx + minx)/2.0f;
 		m->vertexArray[i].y -= (maxy + miny)/2.0f;
 		m->vertexArray[i].z -= (maxz + minz)/2.0f;
 	}
 }
 
 void ScaleModel(Model *m, float sx, float sy, float sz)
 {
 	long i;
 	for (i = 0; i < m->numVertices; i++)
 	{
 		m->vertexArray[i].x *= sx;
 		m->vertexArray[i].y *= sy;
 		m->vertexArray[i].z *= sz;
 	}
 }
 
 static void LOLError(const char *caller, const char *name)
 {
 	static unsigned int draw_error_counter = 0; 
    if(draw_error_counter < NUM_DRAWMODEL_ERROR)
    {
 		    fprintf(stderr, "%s warning: '%s' not found in shader!\n", caller, name);
 		    draw_error_counter++;
    }
    else if(draw_error_counter == NUM_DRAWMODEL_ERROR)
    {
 		    fprintf(stderr, "%s: Number of error bigger than %i. No more vill be printed.\n", caller, NUM_DRAWMODEL_ERROR);
 		    draw_error_counter++;
    }
 }
 
 // This code makes a lot of calls for rebinding variables just in case,
 // and to get attribute locations. This is clearly not optimal, but the
 // goal is stability.
 
 void DrawModel(Model *m, GLuint program, const char* vertexVariableName, const char* normalVariableName, const char* texCoordVariableName)
 {
 	if (m != NULL)
 	{
 		GLint loc;
 		
 		glBindVertexArray(m->vao);	// Select VAO
 		glUseProgram(program); // Added 2022-03
 
 		glBindBuffer(GL_ARRAY_BUFFER, m->vb);
 		loc = glGetAttribLocation(program, vertexVariableName);
 		if (loc >= 0)
 		{
 			glVertexAttribPointer(loc, 3, GL_FLOAT, GL_FALSE, 0, 0); 
 			glEnableVertexAttribArray(loc);
 		}
 		else
 			LOLError("DrawModel", vertexVariableName);
 		
 		if (normalVariableName!=NULL)
 		{
 			loc = glGetAttribLocation(program, normalVariableName);
 			if (loc >= 0)
 			{
 				glBindBuffer(GL_ARRAY_BUFFER, m->nb);
 				glVertexAttribPointer(loc, 3, GL_FLOAT, GL_FALSE, 0, 0);
 				glEnableVertexAttribArray(loc);
 			}
 			else
 				LOLError("DrawModel", normalVariableName);
 		}
 	
 		if ((m->texCoordArray != NULL)&&(texCoordVariableName != NULL))
 		{
 			loc = glGetAttribLocation(program, texCoordVariableName);
 			if (loc >= 0)
 			{
 				glBindBuffer(GL_ARRAY_BUFFER, m->tb);
 				glVertexAttribPointer(loc, 2, GL_FLOAT, GL_FALSE, 0, 0);
 				glEnableVertexAttribArray(loc);
 			}
 			else
 				LOLError("DrawModel", texCoordVariableName);
 		}
 
 		glDrawElements(GL_TRIANGLES, m->numIndices, GL_UNSIGNED_INT, 0L);
 	}
 }
 
 void DrawWireframeModel(Model *m, GLuint program, const char* vertexVariableName, const char* normalVariableName, const char* texCoordVariableName)
 {
 	if (m != NULL)
 	{
 		GLint loc;
 		
 		glBindVertexArray(m->vao);	// Select VAO
 		glUseProgram(program); // Added 2022-03
 
 		glBindBuffer(GL_ARRAY_BUFFER, m->vb);
 		loc = glGetAttribLocation(program, vertexVariableName);
 		if (loc >= 0)
 		{
 			glVertexAttribPointer(loc, 3, GL_FLOAT, GL_FALSE, 0, 0); 
 			glEnableVertexAttribArray(loc);
 		}
 		else
 			LOLError("DrawWireframeModel", vertexVariableName);
 		
 		if (normalVariableName!=NULL)
 		{
 			loc = glGetAttribLocation(program, normalVariableName);
 			if (loc >= 0)
 			{
 				glBindBuffer(GL_ARRAY_BUFFER, m->nb);
 				glVertexAttribPointer(loc, 3, GL_FLOAT, GL_FALSE, 0, 0);
 				glEnableVertexAttribArray(loc);
 			}
 			else
 				LOLError("DrawWireframeModel", normalVariableName);
 		}
 	
 		if ((m->texCoordArray != NULL)&&(texCoordVariableName != NULL))
 		{
 			loc = glGetAttribLocation(program, texCoordVariableName);
 			if (loc >= 0)
 			{
 				glBindBuffer(GL_ARRAY_BUFFER, m->tb);
 				glVertexAttribPointer(loc, 2, GL_FLOAT, GL_FALSE, 0, 0);
 				glEnableVertexAttribArray(loc);
 			}
 			else
 				LOLError("DrawWireframeModel", texCoordVariableName);
 		}
 		glDrawElements(GL_LINE_STRIP, m->numIndices, GL_UNSIGNED_INT, 0L);
 	}
 }
 	
 // Called from LoadModel, LoadModelSet and LoadDataToModel
 // VAO and VBOs must already exist!
 // Useful by its own when the model changes on CPU
 void ReloadModelData(Model *m)
 {
 	glBindVertexArray(m->vao);
 	
 	// VBO for vertex data
 	glBindBuffer(GL_ARRAY_BUFFER, m->vb);
 	glBufferData(GL_ARRAY_BUFFER, m->numVertices*3*sizeof(GLfloat), m->vertexArray, GL_STATIC_DRAW);
 	
 	// VBO for normal data
 	glBindBuffer(GL_ARRAY_BUFFER, m->nb);
 	glBufferData(GL_ARRAY_BUFFER, m->numVertices*3*sizeof(GLfloat), m->normalArray, GL_STATIC_DRAW);
 	
 	// VBO for texture coordinate data NEW for 5b
 	if (m->texCoordArray != NULL)
 	{
 		glBindBuffer(GL_ARRAY_BUFFER, m->tb);
 		glBufferData(GL_ARRAY_BUFFER, m->numVertices*2*sizeof(GLfloat), m->texCoordArray, GL_STATIC_DRAW);
 	}
 	
 	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m->ib);
 	glBufferData(GL_ELEMENT_ARRAY_BUFFER, m->numIndices*sizeof(GLuint), m->indexArray, GL_STATIC_DRAW);
 }
 
 static void GenModelBuffers(Model *m)
 {
 	glGenVertexArrays(1, &m->vao);
 	glGenBuffers(1, &m->vb);
 	glGenBuffers(1, &m->ib);
 	glGenBuffers(1, &m->nb);
 	if (m->texCoordArray != NULL)
 		glGenBuffers(1, &m->tb);
 
 	ReloadModelData(m);
 }
 
 // For simple models
 Model* LoadModel(const char* name)
 {
 	Model* model = NULL;
 	Mesh* mesh = LoadOBJ(name);
 	DecomposeToTriangles(mesh);
 	GenerateNormals(mesh);
 	model = GenerateModel(mesh);
 	DisposeMesh(mesh);
 
 	GenModelBuffers(model);
 	model->externalData = 0;
 	return model;
 }
 
 // For multiple part models
 Model** LoadModelSet(const char* name)
 {
 	Mesh* mesh = LoadOBJ(name);
 	Mesh **mm = SplitToMeshes(mesh);
 	int i, ii;
 	for (i = 0; mm[i] != NULL; i++) {}
 	Model **md = (Model **)calloc(sizeof(Model *), i+1);
 	for (ii = 0; mm[ii] != NULL; ii++)
 	{
 		DecomposeToTriangles(mm[ii]);
 		GenerateNormals(mm[ii]);
 		md[ii] = GenerateModel(mm[ii]);
 		DisposeMesh(mm[ii]);
 	}
 	free(mm);
 	DisposeMtlList(gMaterials);
 	DisposeMesh(mesh);
 	gMtlNameList = NULL;
 	gMaterials = NULL;
 	
 	for (int i = 0; md[i] != NULL; i++)
 	{
 		GenModelBuffers(md[i]);
 		md[i]->externalData = 0;
 	}
 
 	return md;
 }
 
 // Loader for inline data to Model (almost same as LoadModelPlus)
 Model* LoadDataToModel(
 			vec3 *vertices,
 			vec3 *normals,
 			vec2 *texCoords,
 			vec3 *colors,
 			GLuint *indices,
 			int numVert,
 			int numInd)
 {
 	Model* m = (Model *)malloc(sizeof(Model));
 	memset(m, 0, sizeof(Model));
 	
 	m->vertexArray = vertices;
 	m->texCoordArray = texCoords;
 	m->normalArray = normals;
 	m->indexArray = indices;
 	m->numVertices = numVert;
 	m->numIndices = numInd;
 	m->externalData = 1;
 	
 	GenModelBuffers(m);
 	
 	return m;
 }
 
 // Cleanup function, not tested!
 void DisposeModel(Model *m)
 {
 	if (m != NULL)
 	{
 		if (m->externalData == 0)
 		{
 			if (m->vertexArray != NULL)
 				free(m->vertexArray);
 			if (m->normalArray != NULL)
 				free(m->normalArray);
 			if (m->texCoordArray != NULL)
 				free(m->texCoordArray);
 			if (m->colorArray != NULL) // obsolete?
 				free(m->colorArray);
 			if (m->indexArray != NULL)
 				free(m->indexArray);
 		}
 		
 		// Lazy error checking here since "glDeleteBuffers silently ignores 0's and names that do not correspond to existing buffer objects."
 		glDeleteBuffers(1, &m->vb);
 		glDeleteBuffers(1, &m->ib);
 		glDeleteBuffers(1, &m->nb);
 		glDeleteBuffers(1, &m->tb);
 		glDeleteVertexArrays(1, &m->vao);
 		
 		if (m->material != NULL)
 			free(m->material);
 	}
 	free(m);
 }
 
 #endif
49ca15c4
 #endif