| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version.  | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 1990-1998 NeoGeo BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributors: 2004/2005 Blender Foundation, full recode | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 19:31:27 +00:00
										 |  |  | /** \file blender/render/intern/raytrace/rayobject_octree.cpp
 | 
					
						
							|  |  |  |  *  \ingroup render | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | /* IMPORTANT NOTE: this code must be independent of any other render code
 | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  |  * to use it outside the renderer! */ | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <math.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <float.h>
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | #include <assert.h>
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_material_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 20:43:45 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | #include "rayintersection.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | #include "rayobject.h"
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ********** structs *************** */ | 
					
						
							|  |  |  | #define BRANCH_ARRAY 1024
 | 
					
						
							|  |  |  | #define NODE_ARRAY 4096
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | typedef struct Branch { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	struct Branch *b[8]; | 
					
						
							|  |  |  | } Branch; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | typedef struct OcVal { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	short ocx, ocy, ocz; | 
					
						
							|  |  |  | } OcVal; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | typedef struct Node { | 
					
						
							| 
									
										
										
										
											2009-07-03 18:16:59 +00:00
										 |  |  | 	struct RayFace *v[8]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	struct OcVal ov[8]; | 
					
						
							|  |  |  | 	struct Node *next; | 
					
						
							|  |  |  | } Node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct Octree { | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	RayObject rayobj; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	struct Branch **adrbranch; | 
					
						
							|  |  |  | 	struct Node **adrnode; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	float ocsize;   /* ocsize: mult factor,  max size octree */ | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	float ocfacx, ocfacy, ocfacz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	float min[3], max[3]; | 
					
						
							|  |  |  | 	int ocres; | 
					
						
							|  |  |  | 	int branchcount, nodecount; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	/* during building only */ | 
					
						
							|  |  |  | 	char *ocface;  | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-07-03 18:16:59 +00:00
										 |  |  | 	RayFace **ro_nodes; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	int ro_nodes_size, ro_nodes_used; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | } Octree; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | static int  RE_rayobject_octree_intersect(RayObject *o, Isect *isec); | 
					
						
							|  |  |  | static void RE_rayobject_octree_add(RayObject *o, RayObject *ob); | 
					
						
							|  |  |  | static void RE_rayobject_octree_done(RayObject *o); | 
					
						
							|  |  |  | static void RE_rayobject_octree_free(RayObject *o); | 
					
						
							|  |  |  | static void RE_rayobject_octree_bb(RayObject *o, float *min, float *max); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-01 18:30:59 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This function is not expected to be called by current code state. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-01-14 10:08:47 +00:00
										 |  |  | static float RE_rayobject_octree_cost(RayObject *UNUSED(o)) | 
					
						
							| 
									
										
										
										
											2009-10-01 18:30:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return 1.0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-14 10:08:47 +00:00
										 |  |  | static void RE_rayobject_octree_hint_bb(RayObject *UNUSED(o), RayHint *UNUSED(hint), | 
					
						
							|  |  |  |                                         float *UNUSED(min), float *UNUSED(max)) | 
					
						
							| 
									
										
										
										
											2009-10-01 18:30:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | static RayObjectAPI octree_api = | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | 	RE_rayobject_octree_intersect, | 
					
						
							|  |  |  | 	RE_rayobject_octree_add, | 
					
						
							|  |  |  | 	RE_rayobject_octree_done, | 
					
						
							|  |  |  | 	RE_rayobject_octree_free, | 
					
						
							| 
									
										
										
										
											2009-10-01 18:30:59 +00:00
										 |  |  | 	RE_rayobject_octree_bb, | 
					
						
							|  |  |  | 	RE_rayobject_octree_cost, | 
					
						
							|  |  |  | 	RE_rayobject_octree_hint_bb | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* **************** ocval method ******************* */ | 
					
						
							|  |  |  | /* within one octree node, a set of 3x15 bits defines a 'boundbox' to OR with */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | #define OCVALRES    15
 | 
					
						
							|  |  |  | #define BROW16(min, max)  (((max) >= OCVALRES ? 0xFFFF : (1 << (max + 1)) - 1) - ((min > 0) ? ((1 << (min)) - 1) : 0))
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void calc_ocval_face(float *v1, float *v2, float *v3, float *v4, short x, short y, short z, OcVal *ov) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float min[3], max[3]; | 
					
						
							|  |  |  | 	int ocmin, ocmax; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | 	copy_v3_v3(min, v1); | 
					
						
							|  |  |  | 	copy_v3_v3(max, v1); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	DO_MINMAX(v2, min, max); | 
					
						
							|  |  |  | 	DO_MINMAX(v3, min, max); | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (v4) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		DO_MINMAX(v4, min, max); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ocmin = OCVALRES * (min[0] - x); | 
					
						
							|  |  |  | 	ocmax = OCVALRES * (max[0] - x); | 
					
						
							|  |  |  | 	ov->ocx = BROW16(ocmin, ocmax); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ocmin = OCVALRES * (min[1] - y); | 
					
						
							|  |  |  | 	ocmax = OCVALRES * (max[1] - y); | 
					
						
							|  |  |  | 	ov->ocy = BROW16(ocmin, ocmax); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ocmin = OCVALRES * (min[2] - z); | 
					
						
							|  |  |  | 	ocmax = OCVALRES * (max[2] - z); | 
					
						
							|  |  |  | 	ov->ocz = BROW16(ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void calc_ocval_ray(OcVal *ov, float xo, float yo, float zo, float *vec1, float *vec2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ocmin, ocmax; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (vec1[0] < vec2[0]) { | 
					
						
							|  |  |  | 		ocmin = OCVALRES * (vec1[0] - xo); | 
					
						
							|  |  |  | 		ocmax = OCVALRES * (vec2[0] - xo); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		ocmin = OCVALRES * (vec2[0] - xo); | 
					
						
							|  |  |  | 		ocmax = OCVALRES * (vec1[0] - xo); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ov->ocx = BROW16(ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (vec1[1] < vec2[1]) { | 
					
						
							|  |  |  | 		ocmin = OCVALRES * (vec1[1] - yo); | 
					
						
							|  |  |  | 		ocmax = OCVALRES * (vec2[1] - yo); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		ocmin = OCVALRES * (vec2[1] - yo); | 
					
						
							|  |  |  | 		ocmax = OCVALRES * (vec1[1] - yo); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ov->ocy = BROW16(ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (vec1[2] < vec2[2]) { | 
					
						
							|  |  |  | 		ocmin = OCVALRES * (vec1[2] - zo); | 
					
						
							|  |  |  | 		ocmax = OCVALRES * (vec2[2] - zo); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		ocmin = OCVALRES * (vec2[2] - zo); | 
					
						
							|  |  |  | 		ocmax = OCVALRES * (vec1[2] - zo); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ov->ocz = BROW16(ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ************* octree ************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Branch *addbranch(Octree *oc, Branch *br, short ocb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (br->b[ocb]) return br->b[ocb]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	oc->branchcount++; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	index = oc->branchcount >> 12; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->adrbranch[index] == NULL) | 
					
						
							|  |  |  | 		oc->adrbranch[index] = (Branch *)MEM_callocN(4096 * sizeof(Branch), "new oc branch"); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->branchcount >= BRANCH_ARRAY * 4096) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		printf("error; octree branches full\n"); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		oc->branchcount = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	return br->b[ocb] = oc->adrbranch[index] + (oc->branchcount & 4095); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Node *addnode(Octree *oc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int index; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	oc->nodecount++; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	index = oc->nodecount >> 12; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->adrnode[index] == NULL) | 
					
						
							|  |  |  | 		oc->adrnode[index] = (Node *)MEM_callocN(4096 * sizeof(Node), "addnode"); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->nodecount > NODE_ARRAY * NODE_ARRAY) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		printf("error; octree nodes full\n"); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		oc->nodecount = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	return oc->adrnode[index] + (oc->nodecount & 4095); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:29:01 +00:00
										 |  |  | static int face_in_node(RayFace *face, short x, short y, short z, float rtf[4][3]) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	static float nor[3], d; | 
					
						
							|  |  |  | 	float fx, fy, fz; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	// init static vars 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (face) { | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		normal_tri_v3(nor, rtf[0], rtf[1], rtf[2]); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		d = -nor[0] * rtf[0][0] - nor[1] * rtf[0][1] - nor[2] * rtf[0][2]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	fx = x; | 
					
						
							|  |  |  | 	fy = y; | 
					
						
							|  |  |  | 	fz = z; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((fx) * nor[0] + (fy) * nor[1] + (fz) * nor[2] + d > 0.0f) { | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy    ) * nor[1] + (fz    ) * nor[2] + d < 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx    ) * nor[0] + (fy + 1) * nor[1] + (fz    ) * nor[2] + d < 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy + 1) * nor[1] + (fz    ) * nor[2] + d < 0.0f) return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if ((fx    ) * nor[0] + (fy    ) * nor[1] + (fz + 1) * nor[2] + d < 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy    ) * nor[1] + (fz + 1) * nor[2] + d < 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx    ) * nor[0] + (fy + 1) * nor[1] + (fz + 1) * nor[2] + d < 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy + 1) * nor[1] + (fz + 1) * nor[2] + d < 0.0f) return 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		if ((fx + 1) * nor[0] + (fy    ) * nor[1] + (fz    ) * nor[2] + d > 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx    ) * nor[0] + (fy + 1) * nor[1] + (fz    ) * nor[2] + d > 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy + 1) * nor[1] + (fz    ) * nor[2] + d > 0.0f) return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if ((fx    ) * nor[0] + (fy    ) * nor[1] + (fz + 1) * nor[2] + d > 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy    ) * nor[1] + (fz + 1) * nor[2] + d > 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx    ) * nor[0] + (fy + 1) * nor[1] + (fz + 1) * nor[2] + d > 0.0f) return 1; | 
					
						
							|  |  |  | 		if ((fx + 1) * nor[0] + (fy + 1) * nor[1] + (fz + 1) * nor[2] + d > 0.0f) return 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-16 06:56:50 +00:00
										 |  |  | static void ocwrite(Octree *oc, RayFace *face, int quad, short x, short y, short z, float rtf[4][3]) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Branch *br; | 
					
						
							|  |  |  | 	Node *no; | 
					
						
							|  |  |  | 	short a, oc0, oc1, oc2, oc3, oc4, oc5; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	x <<= 2; | 
					
						
							|  |  |  | 	y <<= 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	br = oc->adrbranch[0]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->ocres == 512) { | 
					
						
							|  |  |  | 		oc0 = ((x & 1024) + (y & 512) + (z & 256)) >> 8; | 
					
						
							|  |  |  | 		br = addbranch(oc, br, oc0); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->ocres >= 256) { | 
					
						
							|  |  |  | 		oc0 = ((x & 512) + (y & 256) + (z & 128)) >> 7; | 
					
						
							|  |  |  | 		br = addbranch(oc, br, oc0); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->ocres >= 128) { | 
					
						
							|  |  |  | 		oc0 = ((x & 256) + (y & 128) + (z & 64)) >> 6; | 
					
						
							|  |  |  | 		br = addbranch(oc, br, oc0); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc0 = ((x & 128) + (y & 64) + (z & 32)) >> 5; | 
					
						
							|  |  |  | 	oc1 = ((x & 64) + (y & 32) + (z & 16)) >> 4; | 
					
						
							|  |  |  | 	oc2 = ((x & 32) + (y & 16) + (z & 8)) >> 3; | 
					
						
							|  |  |  | 	oc3 = ((x & 16) + (y & 8) + (z & 4)) >> 2; | 
					
						
							|  |  |  | 	oc4 = ((x & 8) + (y & 4) + (z & 2)) >> 1; | 
					
						
							|  |  |  | 	oc5 = ((x & 4) + (y & 2) + (z & 1)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	br = addbranch(oc, br, oc0); | 
					
						
							|  |  |  | 	br = addbranch(oc, br, oc1); | 
					
						
							|  |  |  | 	br = addbranch(oc, br, oc2); | 
					
						
							|  |  |  | 	br = addbranch(oc, br, oc3); | 
					
						
							|  |  |  | 	br = addbranch(oc, br, oc4); | 
					
						
							|  |  |  | 	no = (Node *)br->b[oc5]; | 
					
						
							|  |  |  | 	if (no == NULL) br->b[oc5] = (Branch *)(no = addnode(oc)); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	while (no->next) no = no->next; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	a = 0; | 
					
						
							|  |  |  | 	if (no->v[7]) {     /* node full */ | 
					
						
							|  |  |  | 		no->next = addnode(oc); | 
					
						
							|  |  |  | 		no = no->next; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		while (no->v[a] != NULL) a++; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	no->v[a] = (RayFace *) RE_rayobject_align(face); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (quad) | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		calc_ocval_face(rtf[0], rtf[1], rtf[2], rtf[3], x >> 2, y >> 1, z, &no->ov[a]); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		calc_ocval_face(rtf[0], rtf[1], rtf[2], NULL, x >> 2, y >> 1, z, &no->ov[a]); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:29:01 +00:00
										 |  |  | static void d2dda(Octree *oc, short b1, short b2, short c1, short c2, char *ocface, short rts[4][3], float rtf[4][3]) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	int ocx1, ocx2, ocy1, ocy2; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	int x, y, dx = 0, dy = 0; | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	float ox1, ox2, oy1, oy2; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 	float lambda, lambda_o, lambda_x, lambda_y, ldx, ldy; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ocx1 = rts[b1][c1]; | 
					
						
							|  |  |  | 	ocy1 = rts[b1][c2]; | 
					
						
							|  |  |  | 	ocx2 = rts[b2][c1]; | 
					
						
							|  |  |  | 	ocy2 = rts[b2][c2]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (ocx1 == ocx2 && ocy1 == ocy2) { | 
					
						
							|  |  |  | 		ocface[oc->ocres * ocx1 + ocy1] = 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ox1 = rtf[b1][c1]; | 
					
						
							|  |  |  | 	oy1 = rtf[b1][c2]; | 
					
						
							|  |  |  | 	ox2 = rtf[b2][c1]; | 
					
						
							|  |  |  | 	oy2 = rtf[b2][c2]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (ox1 != ox2) { | 
					
						
							|  |  |  | 		if (ox2 - ox1 > 0.0f) { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_x = (ox1 - ocx1 - 1.0f) / (ox1 - ox2); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldx = -1.0f / (ox1 - ox2); | 
					
						
							|  |  |  | 			dx = 1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_x = (ox1 - ocx1) / (ox1 - ox2); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldx = 1.0f / (ox1 - ox2); | 
					
						
							|  |  |  | 			dx = -1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 		lambda_x = 1.0f; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		ldx = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oy1 != oy2) { | 
					
						
							|  |  |  | 		if (oy2 - oy1 > 0.0f) { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_y = (oy1 - ocy1 - 1.0f) / (oy1 - oy2); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldy = -1.0f / (oy1 - oy2); | 
					
						
							|  |  |  | 			dy = 1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_y = (oy1 - ocy1) / (oy1 - oy2); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldy = 1.0f / (oy1 - oy2); | 
					
						
							|  |  |  | 			dy = -1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 		lambda_y = 1.0f; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		ldy = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	x = ocx1; y = ocy1; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 	lambda = MIN2(lambda_x, lambda_y); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	while (TRUE) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-10-14 13:08:19 +00:00
										 |  |  | 		if (x < 0 || y < 0 || x >= oc->ocres || y >= oc->ocres) { | 
					
						
							|  |  |  | 			/* pass*/ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			ocface[oc->ocres * x + y] = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 		lambda_o = lambda; | 
					
						
							|  |  |  | 		if (lambda_x == lambda_y) { | 
					
						
							|  |  |  | 			lambda_x += ldx; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			x += dx; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_y += ldy; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			y += dy; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			if (lambda_x < lambda_y) { | 
					
						
							|  |  |  | 				lambda_x += ldx; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				x += dx; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 				lambda_y += ldy; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				y += dy; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 		lambda = MIN2(lambda_x, lambda_y); | 
					
						
							|  |  |  | 		if (lambda == lambda_o) break; | 
					
						
							|  |  |  | 		if (lambda >= 1.0f) break; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ocface[oc->ocres * ocx2 + ocy2] = 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | static void filltriangle(Octree *oc, short c1, short c2, char *ocface, short *ocmin, short *ocmax) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int a, x, y, y1, y2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	for (x = ocmin[c1]; x <= ocmax[c1]; x++) { | 
					
						
							|  |  |  | 		a = oc->ocres * x; | 
					
						
							|  |  |  | 		for (y = ocmin[c2]; y <= ocmax[c2]; y++) { | 
					
						
							|  |  |  | 			if (ocface[a + y]) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				y++; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				while (ocface[a + y] && y != ocmax[c2]) y++; | 
					
						
							|  |  |  | 				for (y1 = ocmax[c2]; y1 > y; y1--) { | 
					
						
							|  |  |  | 					if (ocface[a + y1]) { | 
					
						
							|  |  |  | 						for (y2 = y; y2 <= y1; y2++) ocface[a + y2] = 1; | 
					
						
							|  |  |  | 						y1 = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				y = ocmax[c2]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | static void RE_rayobject_octree_free(RayObject *tree) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	Octree *oc = (Octree *)tree; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	printf("branches %d nodes %d\n", oc->branchcount, oc->nodecount); | 
					
						
							| 
									
										
										
										
											2012-03-31 00:59:17 +00:00
										 |  |  | 	printf("raycount %d\n", raycount); | 
					
						
							|  |  |  | 	printf("ray coherent %d\n", coherent_ray); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	printf("accepted %d rejected %d\n", accepted, rejected); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (oc->ocface) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		MEM_freeN(oc->ocface); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (oc->adrbranch) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		int a = 0; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		while (oc->adrbranch[a]) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			MEM_freeN(oc->adrbranch[a]); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			oc->adrbranch[a] = NULL; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			a++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		MEM_freeN(oc->adrbranch); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		oc->adrbranch = NULL; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->branchcount = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (oc->adrnode) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		int a = 0; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		while (oc->adrnode[a]) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			MEM_freeN(oc->adrnode[a]); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			oc->adrnode[a] = NULL; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			a++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		MEM_freeN(oc->adrnode); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		oc->adrnode = NULL; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->nodecount = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(oc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-13 01:56:03 +00:00
										 |  |  | RayObject *RE_rayobject_octree_create(int ocres, int size) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	Octree *oc = (Octree *)MEM_callocN(sizeof(Octree), "Octree"); | 
					
						
							|  |  |  | 	assert(RE_rayobject_isAligned(oc) );  /* RayObject API assumes real data to be 4-byte aligned */ | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	oc->rayobj.api = &octree_api; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	oc->ocres = ocres; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->ro_nodes = (RayFace **)MEM_callocN(sizeof(RayFace *) * size, "octree rayobject nodes"); | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	oc->ro_nodes_size = size; | 
					
						
							|  |  |  | 	oc->ro_nodes_used = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	return RE_rayobject_unalignRayAPI((RayObject *) oc); | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | static void RE_rayobject_octree_add(RayObject *tree, RayObject *node) | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	Octree *oc = (Octree *)tree; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	assert(RE_rayobject_isRayFace(node) ); | 
					
						
							|  |  |  | 	assert(oc->ro_nodes_used < oc->ro_nodes_size); | 
					
						
							|  |  |  | 	oc->ro_nodes[oc->ro_nodes_used++] = (RayFace *)RE_rayobject_align(node); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | static void octree_fill_rayface(Octree *oc, RayFace *face) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	float ocfac[3], rtf[4][3]; | 
					
						
							| 
									
										
											  
											
												
Render Instancing
=================
Big commit, but little user visible changes.
- Dupliverts and duplifaces are now rendered as instances, instead
  of storing all of the geometry for each dupli, now an instance is
  created with a matrix transform refering to the source object.
  This should allow us to render tree leaves more memory efficient.
- Radiosity and to some degree raytracing of such objects is not
  really efficient still. For radiosity this is fundamentally hard
  to solve, but raytracing an octree could be created for each object,
  but the current octree code with it's fixed size doesn't allow this
  efficiently.
- The regression tests survived, but with I expect that some bugs will
  pop up .. hopefully not too many :).
Implementation Notes
====================
- Dupligroups and linked meshes are not rendered as instances yet,
  since they can in fact be different due to various reasons,
  instancing of these types of duplis that are the same can be added
  for them at a later point.
- Each ObjectRen now stores it's own database, instead of there being
  one big databases of faces, verts, .. . Which objects that are actually
  rendered are defined by the list of ObjectRenInstances, which all refer
  to an ObjectRen.
- Homogeneous coordinatess and clipping is now not stored in vertices
  anymore, but instead computed on the fly. This couldn't work for
  instances. That does mean some extra computation has to be done, but
  memory lookups can be slow too, and this saves some memory. Overall
  I didn't find a significant speed impact.
- OSA rendering for solid and ztransp now is different. Instead of e.g.
  going 8 times over the databases times and rendering the z-buffer, it
  now goes over the database once and renders each polygon 8 times. That
  was necessary to keep instances efficient, and can also give some
  performance improvement without instances.
- There was already instancing support in the yafray export code, now it
  uses Blender's render instances for export.
- UV and color layer storage in the render was a bit messy before, now
  should be easier to understand.
- convertblender.c was reorganized somewhat. Regular render, speedvector
  and baking now use a single function to create the database, previously
  there was code duplicated for it.
- Some of these changes were done with future multithreading of scene
  and shadow buffer creation in mind, though especially for scene creation
  much work remains to be done to make it threadsafe, since it also involves
  a lot of code from blenkernel, and there is an ugly conflict with the way
  dupli groups work here .. though in the render code itself it's almost there.
											
										 
											2007-12-15 20:41:45 +00:00
										 |  |  | 	float co1[3], co2[3], co3[3], co4[3]; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	short rts[4][3]; | 
					
						
							|  |  |  | 	short ocmin[3], ocmax[3]; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	char *ocface = oc->ocface;   // front, top, size view of face, to fill in
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	int a, b, c, oc1, oc2, oc3, oc4, x, y, z, ocres2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ocfac[0] = oc->ocfacx; | 
					
						
							|  |  |  | 	ocfac[1] = oc->ocfacy; | 
					
						
							|  |  |  | 	ocfac[2] = oc->ocfacz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ocres2 = oc->ocres * oc->ocres; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | 	copy_v3_v3(co1, face->v1); | 
					
						
							|  |  |  | 	copy_v3_v3(co2, face->v2); | 
					
						
							|  |  |  | 	copy_v3_v3(co3, face->v3); | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (face->v4) | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | 		copy_v3_v3(co4, face->v4); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	for (c = 0; c < 3; c++) { | 
					
						
							| 
									
										
										
										
											2012-02-27 10:35:39 +00:00
										 |  |  | 		rtf[0][c] = (co1[c] - oc->min[c]) * ocfac[c]; | 
					
						
							|  |  |  | 		rts[0][c] = (short)rtf[0][c]; | 
					
						
							|  |  |  | 		rtf[1][c] = (co2[c] - oc->min[c]) * ocfac[c]; | 
					
						
							|  |  |  | 		rts[1][c] = (short)rtf[1][c]; | 
					
						
							|  |  |  | 		rtf[2][c] = (co3[c] - oc->min[c]) * ocfac[c]; | 
					
						
							|  |  |  | 		rts[2][c] = (short)rtf[2][c]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (RE_rayface_isQuad(face)) { | 
					
						
							| 
									
										
										
										
											2012-02-27 10:35:39 +00:00
										 |  |  | 			rtf[3][c] = (co4[c] - oc->min[c]) * ocfac[c]; | 
					
						
							|  |  |  | 			rts[3][c] = (short)rtf[3][c]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	for (c = 0; c < 3; c++) { | 
					
						
							|  |  |  | 		oc1 = rts[0][c]; | 
					
						
							|  |  |  | 		oc2 = rts[1][c]; | 
					
						
							|  |  |  | 		oc3 = rts[2][c]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (!RE_rayface_isQuad(face)) { | 
					
						
							| 
									
										
										
										
											2012-12-21 05:07:26 +00:00
										 |  |  | 			ocmin[c] = min_iii(oc1, oc2, oc3); | 
					
						
							|  |  |  | 			ocmax[c] = max_iii(oc1, oc2, oc3); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			oc4 = rts[3][c]; | 
					
						
							| 
									
										
										
										
											2012-12-21 05:07:26 +00:00
										 |  |  | 			ocmin[c] = min_iiii(oc1, oc2, oc3, oc4); | 
					
						
							|  |  |  | 			ocmax[c] = max_iiii(oc1, oc2, oc3, oc4); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		if (ocmax[c] > oc->ocres - 1) ocmax[c] = oc->ocres - 1; | 
					
						
							|  |  |  | 		if (ocmin[c] < 0) ocmin[c] = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (ocmin[0] == ocmax[0] && ocmin[1] == ocmax[1] && ocmin[2] == ocmax[2]) { | 
					
						
							| 
									
										
										
										
											2009-08-05 15:50:57 +00:00
										 |  |  | 		ocwrite(oc, face, RE_rayface_isQuad(face), ocmin[0], ocmin[1], ocmin[2], rtf); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		d2dda(oc, 0, 1, 0, 1, ocface + ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		d2dda(oc, 0, 1, 0, 2, ocface, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		d2dda(oc, 0, 1, 1, 2, ocface + 2 * ocres2, rts, rtf); | 
					
						
							|  |  |  | 		d2dda(oc, 1, 2, 0, 1, ocface + ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		d2dda(oc, 1, 2, 0, 2, ocface, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		d2dda(oc, 1, 2, 1, 2, ocface + 2 * ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (!RE_rayface_isQuad(face)) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			d2dda(oc, 2, 0, 0, 1, ocface + ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 			d2dda(oc, 2, 0, 0, 2, ocface, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			d2dda(oc, 2, 0, 1, 2, ocface + 2 * ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			d2dda(oc, 2, 3, 0, 1, ocface + ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 			d2dda(oc, 2, 3, 0, 2, ocface, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			d2dda(oc, 2, 3, 1, 2, ocface + 2 * ocres2, rts, rtf); | 
					
						
							|  |  |  | 			d2dda(oc, 3, 0, 0, 1, ocface + ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 			d2dda(oc, 3, 0, 0, 2, ocface, rts, rtf); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			d2dda(oc, 3, 0, 1, 2, ocface + 2 * ocres2, rts, rtf); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		/* nothing todo with triangle..., just fills :) */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		filltriangle(oc, 0, 1, ocface + ocres2, ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		filltriangle(oc, 0, 2, ocface, ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		filltriangle(oc, 1, 2, ocface + 2 * ocres2, ocmin, ocmax); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* init static vars here */ | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		face_in_node(face, 0, 0, 0, rtf); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		for (x = ocmin[0]; x <= ocmax[0]; x++) { | 
					
						
							|  |  |  | 			a = oc->ocres * x; | 
					
						
							|  |  |  | 			for (y = ocmin[1]; y <= ocmax[1]; y++) { | 
					
						
							|  |  |  | 				if (ocface[a + y + ocres2]) { | 
					
						
							|  |  |  | 					b = oc->ocres * y + 2 * ocres2; | 
					
						
							|  |  |  | 					for (z = ocmin[2]; z <= ocmax[2]; z++) { | 
					
						
							|  |  |  | 						if (ocface[b + z] && ocface[a + z]) { | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 							if (face_in_node(NULL, x, y, z, rtf)) | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 								ocwrite(oc, face, RE_rayface_isQuad(face), x, y, z, rtf); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* same loops to clear octree, doubt it can be done smarter */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		for (x = ocmin[0]; x <= ocmax[0]; x++) { | 
					
						
							|  |  |  | 			a = oc->ocres * x; | 
					
						
							|  |  |  | 			for (y = ocmin[1]; y <= ocmax[1]; y++) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				/* x-y */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				ocface[a + y + ocres2] = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				b = oc->ocres * y + 2 * ocres2; | 
					
						
							|  |  |  | 				for (z = ocmin[2]; z <= ocmax[2]; z++) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					/* y-z */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 					ocface[b + z] = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					/* x-z */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 					ocface[a + z] = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | static void RE_rayobject_octree_done(RayObject *tree) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	Octree *oc = (Octree *)tree; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	int c; | 
					
						
							|  |  |  | 	float t00, t01, t02; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	int ocres2 = oc->ocres * oc->ocres; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	INIT_MINMAX(oc->min, oc->max); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	/* Calculate Bounding Box */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	for (c = 0; c < oc->ro_nodes_used; c++) | 
					
						
							|  |  |  | 		RE_rayobject_merge_bb(RE_rayobject_unalignRayFace(oc->ro_nodes[c]), oc->min, oc->max); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	/* Alloc memory */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->adrbranch = (Branch **)MEM_callocN(sizeof(void *) * BRANCH_ARRAY, "octree branches"); | 
					
						
							|  |  |  | 	oc->adrnode = (Node **)MEM_callocN(sizeof(void *) * NODE_ARRAY, "octree nodes"); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->adrbranch[0] = (Branch *)MEM_callocN(4096 * sizeof(Branch), "makeoctree"); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	/* the lookup table, per face, for which nodes to fill in */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->ocface = (char *)MEM_callocN(3 * ocres2 + 8, "ocface"); | 
					
						
							|  |  |  | 	memset(oc->ocface, 0, 3 * ocres2); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	for (c = 0; c < 3; c++) { /* octree enlarge, still needed? */ | 
					
						
							|  |  |  | 		oc->min[c] -= 0.01f; | 
					
						
							|  |  |  | 		oc->max[c] += 0.01f; | 
					
						
							| 
									
										
											  
											
												
Render Instancing
=================
Big commit, but little user visible changes.
- Dupliverts and duplifaces are now rendered as instances, instead
  of storing all of the geometry for each dupli, now an instance is
  created with a matrix transform refering to the source object.
  This should allow us to render tree leaves more memory efficient.
- Radiosity and to some degree raytracing of such objects is not
  really efficient still. For radiosity this is fundamentally hard
  to solve, but raytracing an octree could be created for each object,
  but the current octree code with it's fixed size doesn't allow this
  efficiently.
- The regression tests survived, but with I expect that some bugs will
  pop up .. hopefully not too many :).
Implementation Notes
====================
- Dupligroups and linked meshes are not rendered as instances yet,
  since they can in fact be different due to various reasons,
  instancing of these types of duplis that are the same can be added
  for them at a later point.
- Each ObjectRen now stores it's own database, instead of there being
  one big databases of faces, verts, .. . Which objects that are actually
  rendered are defined by the list of ObjectRenInstances, which all refer
  to an ObjectRen.
- Homogeneous coordinatess and clipping is now not stored in vertices
  anymore, but instead computed on the fly. This couldn't work for
  instances. That does mean some extra computation has to be done, but
  memory lookups can be slow too, and this saves some memory. Overall
  I didn't find a significant speed impact.
- OSA rendering for solid and ztransp now is different. Instead of e.g.
  going 8 times over the databases times and rendering the z-buffer, it
  now goes over the database once and renders each polygon 8 times. That
  was necessary to keep instances efficient, and can also give some
  performance improvement without instances.
- There was already instancing support in the yafray export code, now it
  uses Blender's render instances for export.
- UV and color layer storage in the render was a bit messy before, now
  should be easier to understand.
- convertblender.c was reorganized somewhat. Regular render, speedvector
  and baking now use a single function to create the database, previously
  there was code duplicated for it.
- Some of these changes were done with future multithreading of scene
  and shadow buffer creation in mind, though especially for scene creation
  much work remains to be done to make it threadsafe, since it also involves
  a lot of code from blenkernel, and there is an ugly conflict with the way
  dupli groups work here .. though in the render code itself it's almost there.
											
										 
											2007-12-15 20:41:45 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	t00 = oc->max[0] - oc->min[0]; | 
					
						
							|  |  |  | 	t01 = oc->max[1] - oc->min[1]; | 
					
						
							|  |  |  | 	t02 = oc->max[2] - oc->min[2]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	/* this minus 0.1 is old safety... seems to be needed? */ | 
					
						
							| 
									
										
										
										
											2012-07-29 18:14:20 +00:00
										 |  |  | 	oc->ocfacx = (oc->ocres - 0.1f) / t00; | 
					
						
							|  |  |  | 	oc->ocfacy = (oc->ocres - 0.1f) / t01; | 
					
						
							|  |  |  | 	oc->ocfacz = (oc->ocres - 0.1f) / t02; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc->ocsize = sqrt(t00 * t00 + t01 * t01 + t02 * t02);  /* global, max size octree */ | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	for (c = 0; c < oc->ro_nodes_used; c++) { | 
					
						
							| 
									
										
										
										
											2009-07-03 18:16:59 +00:00
										 |  |  | 		octree_fill_rayface(oc, oc->ro_nodes[c]); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	MEM_freeN(oc->ocface); | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 	oc->ocface = NULL; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	MEM_freeN(oc->ro_nodes); | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 	oc->ro_nodes = NULL; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	printf("%f %f - %f\n", oc->min[0], oc->max[0], oc->ocfacx); | 
					
						
							|  |  |  | 	printf("%f %f - %f\n", oc->min[1], oc->max[1], oc->ocfacy); | 
					
						
							|  |  |  | 	printf("%f %f - %f\n", oc->min[2], oc->max[2], oc->ocfacz); | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | static void RE_rayobject_octree_bb(RayObject *tree, float *min, float *max) | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	Octree *oc = (Octree *)tree; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	DO_MINMAX(oc->min, min, max); | 
					
						
							|  |  |  | 	DO_MINMAX(oc->max, min, max); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* check all faces in this node */ | 
					
						
							| 
									
										
										
										
											2012-01-14 10:08:47 +00:00
										 |  |  | static int testnode(Octree *UNUSED(oc), Isect *is, Node *no, OcVal ocval) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	short nr = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* return on any first hit */ | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	if (is->mode == RE_RAY_SHADOW) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		for (; no; no = no->next) { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			for (nr = 0; nr < 8; nr++) { | 
					
						
							|  |  |  | 				RayFace *face = no->v[nr]; | 
					
						
							|  |  |  | 				OcVal     *ov = no->ov + nr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (!face) break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if ( (ov->ocx & ocval.ocx) && (ov->ocy & ocval.ocy) && (ov->ocz & ocval.ocz) ) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 					if (RE_rayobject_intersect(RE_rayobject_unalignRayFace(face), is) ) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 						return 1; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		/* else mirror or glass or shadowtra, return closest face  */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		int found = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		for (; no; no = no->next) { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			for (nr = 0; nr < 8; nr++) { | 
					
						
							|  |  |  | 				RayFace *face = no->v[nr]; | 
					
						
							|  |  |  | 				OcVal     *ov = no->ov + nr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (!face) break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if ( (ov->ocx & ocval.ocx) && (ov->ocy & ocval.ocy) && (ov->ocz & ocval.ocz) ) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 					if (RE_rayobject_intersect(RE_rayobject_unalignRayFace(face), is) ) { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 						found = 1; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		return found; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* find the Node for the octree coord x y z */ | 
					
						
							|  |  |  | static Node *ocread(Octree *oc, int x, int y, int z) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Branch *br; | 
					
						
							|  |  |  | 	int oc1; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	x <<= 2; | 
					
						
							|  |  |  | 	y <<= 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	br = oc->adrbranch[0]; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->ocres == 512) { | 
					
						
							|  |  |  | 		oc1 = ((x & 1024) + (y & 512) + (z & 256)) >> 8; | 
					
						
							|  |  |  | 		br = br->b[oc1]; | 
					
						
							|  |  |  | 		if (br == NULL) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->ocres >= 256) { | 
					
						
							|  |  |  | 		oc1 = ((x & 512) + (y & 256) + (z & 128)) >> 7; | 
					
						
							|  |  |  | 		br = br->b[oc1]; | 
					
						
							|  |  |  | 		if (br == NULL) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->ocres >= 128) { | 
					
						
							|  |  |  | 		oc1 = ((x & 256) + (y & 128) + (z & 64)) >> 6; | 
					
						
							|  |  |  | 		br = br->b[oc1]; | 
					
						
							|  |  |  | 		if (br == NULL) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	oc1 = ((x & 128) + (y & 64) + (z & 32)) >> 5; | 
					
						
							|  |  |  | 	br = br->b[oc1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (br) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		oc1 = ((x & 64) + (y & 32) + (z & 16)) >> 4; | 
					
						
							|  |  |  | 		br = br->b[oc1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (br) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			oc1 = ((x & 32) + (y & 16) + (z & 8)) >> 3; | 
					
						
							|  |  |  | 			br = br->b[oc1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 			if (br) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				oc1 = ((x & 16) + (y & 8) + (z & 4)) >> 2; | 
					
						
							|  |  |  | 				br = br->b[oc1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 				if (br) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 					oc1 = ((x & 8) + (y & 4) + (z & 2)) >> 1; | 
					
						
							|  |  |  | 					br = br->b[oc1]; | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 					if (br) { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 						oc1 = ((x & 4) + (y & 2) + (z & 1)); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 						return (Node *)br->b[oc1]; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int cliptest(float p, float q, float *u1, float *u2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float r; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (p < 0.0f) { | 
					
						
							|  |  |  | 		if (q < p) return 0; | 
					
						
							|  |  |  | 		else if (q < 0.0f) { | 
					
						
							|  |  |  | 			r = q / p; | 
					
						
							|  |  |  | 			if (r > *u2) return 0; | 
					
						
							|  |  |  | 			else if (r > *u1) *u1 = r; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		if (p > 0.0f) { | 
					
						
							|  |  |  | 			if (q < 0.0f) return 0; | 
					
						
							|  |  |  | 			else if (q < p) { | 
					
						
							|  |  |  | 				r = q / p; | 
					
						
							|  |  |  | 				if (r < *u1) return 0; | 
					
						
							|  |  |  | 				else if (r < *u2) *u2 = r; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		else if (q < 0.0f) return 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* extensive coherence checks/storage cancels out the benefit of it, and gives errors... we
 | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  |  * need better methods, sample code commented out below (ton) */ | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  |   | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | in top : static int coh_nodes[16 * 16 * 16][6]; | 
					
						
							|  |  |  | in makeoctree : memset(coh_nodes, 0, sizeof(coh_nodes)); | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  |   | 
					
						
							|  |  |  | static void add_coherence_test(int ocx1, int ocx2, int ocy1, int ocy2, int ocz1, int ocz2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	short *sp; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	sp = coh_nodes[(ocx2 & 15) + 16 * (ocy2 & 15) + 256 * (ocz2 & 15)]; | 
					
						
							|  |  |  | 	sp[0] = ocx1; sp[1] = ocy1; sp[2] = ocz1; | 
					
						
							|  |  |  | 	sp[3] = ocx2; sp[4] = ocy2; sp[5] = ocz2; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int do_coherence_test(int ocx1, int ocx2, int ocy1, int ocy2, int ocz1, int ocz2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	short *sp; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	sp = coh_nodes[(ocx2 & 15) + 16 * (ocy2 & 15) + 256 * (ocz2 & 15)]; | 
					
						
							|  |  |  | 	if (sp[0] == ocx1 && sp[1] == ocy1 && sp[2] == ocz1 && | 
					
						
							|  |  |  | 	    sp[3] == ocx2 && sp[4] == ocy2 && sp[5] == ocz2) return 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* return 1: found valid intersection */ | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | /* starts with is->orig.face */ | 
					
						
							| 
									
										
										
										
											2009-08-25 20:26:50 +00:00
										 |  |  | static int RE_rayobject_octree_intersect(RayObject *tree, Isect *is) | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	Octree *oc = (Octree *)tree; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	Node *no; | 
					
						
							|  |  |  | 	OcVal ocval; | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 	float vec1[3], vec2[3], start[3], end[3]; | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	float u1, u2, ox1, ox2, oy1, oy2, oz1, oz2; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 	float lambda_o, lambda_x, ldx, lambda_y, ldy, lambda_z, ldz, dda_lambda; | 
					
						
							|  |  |  | 	float o_lambda = 0; | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	int dx, dy, dz; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	int xo, yo, zo, c1 = 0; | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	int ocx1, ocx2, ocy1, ocy2, ocz1, ocz2; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* clip with octree */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (oc->branchcount == 0) return 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* do this before intersect calls */ | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	is->facecontr = NULL;                /* to check shared edge */ | 
					
						
							|  |  |  | 	is->obcontr = 0; | 
					
						
							|  |  |  | 	is->faceisect = is->isect = 0;        /* shared edge, quad half flag */ | 
					
						
							|  |  |  | 	is->userdata = oc->userdata; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	copy_v3_v3(start, is->start); | 
					
						
							|  |  |  | 	madd_v3_v3v3fl(end, is->start, is->dir, is->dist); | 
					
						
							|  |  |  | 	ldx = is->dir[0] * is->dist; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 	o_lambda = is->dist; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	u1 = 0.0f; | 
					
						
							|  |  |  | 	u2 = 1.0f; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 	/* clip with octree cube */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (cliptest(-ldx, start[0] - oc->min[0], &u1, &u2)) { | 
					
						
							|  |  |  | 		if (cliptest(ldx, oc->max[0] - start[0], &u1, &u2)) { | 
					
						
							|  |  |  | 			ldy = is->dir[1] * is->dist; | 
					
						
							|  |  |  | 			if (cliptest(-ldy, start[1] - oc->min[1], &u1, &u2)) { | 
					
						
							|  |  |  | 				if (cliptest(ldy, oc->max[1] - start[1], &u1, &u2)) { | 
					
						
							|  |  |  | 					ldz = is->dir[2] * is->dist; | 
					
						
							|  |  |  | 					if (cliptest(-ldz, start[2] - oc->min[2], &u1, &u2)) { | 
					
						
							|  |  |  | 						if (cliptest(ldz, oc->max[2] - start[2], &u1, &u2)) { | 
					
						
							|  |  |  | 							c1 = 1; | 
					
						
							|  |  |  | 							if (u2 < 1.0f) { | 
					
						
							|  |  |  | 								end[0] = start[0] + u2 * ldx; | 
					
						
							|  |  |  | 								end[1] = start[1] + u2 * ldy; | 
					
						
							|  |  |  | 								end[2] = start[2] + u2 * ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 							} | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 							if (u1 > 0.0f) { | 
					
						
							|  |  |  | 								start[0] += u1 * ldx; | 
					
						
							|  |  |  | 								start[1] += u1 * ldy; | 
					
						
							|  |  |  | 								start[2] += u1 * ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 							} | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (c1 == 0) return 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* reset static variables in ocread */ | 
					
						
							|  |  |  | 	//ocread(oc, oc->ocres, 0, 0);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* setup 3dda to traverse octree */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	ox1 = (start[0] - oc->min[0]) * oc->ocfacx; | 
					
						
							|  |  |  | 	oy1 = (start[1] - oc->min[1]) * oc->ocfacy; | 
					
						
							|  |  |  | 	oz1 = (start[2] - oc->min[2]) * oc->ocfacz; | 
					
						
							|  |  |  | 	ox2 = (end[0] - oc->min[0]) * oc->ocfacx; | 
					
						
							|  |  |  | 	oy2 = (end[1] - oc->min[1]) * oc->ocfacy; | 
					
						
							|  |  |  | 	oz2 = (end[2] - oc->min[2]) * oc->ocfacz; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ocx1 = (int)ox1; | 
					
						
							|  |  |  | 	ocy1 = (int)oy1; | 
					
						
							|  |  |  | 	ocz1 = (int)oz1; | 
					
						
							|  |  |  | 	ocx2 = (int)ox2; | 
					
						
							|  |  |  | 	ocy2 = (int)oy2; | 
					
						
							|  |  |  | 	ocz2 = (int)oz2; | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 	if (ocx1 == ocx2 && ocy1 == ocy2 && ocz1 == ocz2) { | 
					
						
							|  |  |  | 		no = ocread(oc, ocx1, ocy1, ocz1); | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (no) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			/* exact intersection with node */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			vec1[0] = ox1; vec1[1] = oy1; vec1[2] = oz1; | 
					
						
							|  |  |  | 			vec2[0] = ox2; vec2[1] = oy2; vec2[2] = oz2; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			calc_ocval_ray(&ocval, (float)ocx1, (float)ocy1, (float)ocz1, vec1, vec2); | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			if (testnode(oc, is, no, ocval) ) return 1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 		int found = 0; | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		//static int coh_ocx1, coh_ocx2, coh_ocy1, coh_ocy2, coh_ocz1, coh_ocz2;
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		float dox, doy, doz; | 
					
						
							|  |  |  | 		int eqval; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 		/* calc lambda en ld */ | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		dox = ox1 - ox2; | 
					
						
							|  |  |  | 		doy = oy1 - oy2; | 
					
						
							|  |  |  | 		doz = oz1 - oz2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (dox < -FLT_EPSILON) { | 
					
						
							|  |  |  | 			ldx = -1.0f / dox; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_x = (ocx1 - ox1 + 1.0f) * ldx; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			dx = 1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		else if (dox > FLT_EPSILON) { | 
					
						
							|  |  |  | 			ldx = 1.0f / dox; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_x = (ox1 - ocx1) * ldx; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			dx = -1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_x = 1.0f; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldx = 0; | 
					
						
							|  |  |  | 			dx = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		if (doy < -FLT_EPSILON) { | 
					
						
							|  |  |  | 			ldy = -1.0f / doy; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_y = (ocy1 - oy1 + 1.0f) * ldy; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			dy = 1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		else if (doy > FLT_EPSILON) { | 
					
						
							|  |  |  | 			ldy = 1.0f / doy; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_y = (oy1 - ocy1) * ldy; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			dy = -1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_y = 1.0f; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldy = 0; | 
					
						
							|  |  |  | 			dy = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		if (doz < -FLT_EPSILON) { | 
					
						
							|  |  |  | 			ldz = -1.0f / doz; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_z = (ocz1 - oz1 + 1.0f) * ldz; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			dz = 1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		else if (doz > FLT_EPSILON) { | 
					
						
							|  |  |  | 			ldz = 1.0f / doz; | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_z = (oz1 - ocz1) * ldz; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			dz = -1; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_z = 1.0f; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			ldz = 0; | 
					
						
							|  |  |  | 			dz = 0; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		xo = ocx1; yo = ocy1; zo = ocz1; | 
					
						
							| 
									
										
										
										
											2012-12-21 05:07:26 +00:00
										 |  |  | 		dda_lambda = min_fff(lambda_x, lambda_y, lambda_z); | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 		vec2[0] = ox1; | 
					
						
							|  |  |  | 		vec2[1] = oy1; | 
					
						
							|  |  |  | 		vec2[2] = oz1; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		/* this loop has been constructed to make sure the first and last node of ray
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 		 * are always included, even when dda_lambda==1.0f or larger */ | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		while (TRUE) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			no = ocread(oc, xo, yo, zo); | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 			if (no) { | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				 | 
					
						
							|  |  |  | 				/* calculate ray intersection with octree node */ | 
					
						
							| 
									
										
											  
											
												Raytrace modifications from the Render Branch.
These should not have any effect on render results, except in some cases with
you have overlapping faces, where the noise seems to be slightly reduced.
There are some performance improvements, for simple scenes I wouldn't expect
more than 5-10% to be cut off the render time, for sintel scenes we got about
50% on average, that's with millions of polygons on intel quad cores. This
because memory access / cache misses were the main bottleneck for those scenes,
and the optimizations improve that.
Interal changes:
* Remove RE_raytrace.h, raytracer is now only used by render engine again.
* Split non-public parts rayobject.h into rayobject_internal.h, hopefully
 makes it clearer how the API is used.
* Added rayintersection.h to contain some of the stuff from RE_raytrace.h
* Change Isect.vec/labda to Isect.dir/dist, previously vec was sometimes
  normalized and sometimes not, confusing... now dir is always normalized
  and dist contains the distance.
* Change VECCOPY and similar to BLI_math functions.
* Force inlining of auxiliary functions for ray-triangle/quad intersection,
  helps a few percentages.
* Reorganize svbvh code so all the traversal functions are in one file
* Don't do test for root so that push_childs can be inlined
* Make shadow a template parameter so it doesn't need to be runtime checked
* Optimization in raytree building, was computing bounding boxes more often
  than necessary.
* Leave out logf() factor in SAH, makes tree build quicker with no
  noticeable influence on raytracing on performance?
* Set max childs to 4, simplifies traversal code a bit, but also seems
  to help slightly in general.
* Store child pointers and child bb just as fixed arrays of size 4 in nodes,
  nearly all nodes have this many children, so overall it actually reduces
  memory usage a bit and avoids a pointer indirection.
											
										 
											2011-02-05 13:41:29 +00:00
										 |  |  | 				copy_v3_v3(vec1, vec2); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 				// dox, y, z is negative
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 				vec2[0] = ox1 - dda_lambda * dox; | 
					
						
							|  |  |  | 				vec2[1] = oy1 - dda_lambda * doy; | 
					
						
							|  |  |  | 				vec2[2] = oz1 - dda_lambda * doz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				calc_ocval_ray(&ocval, (float)xo, (float)yo, (float)zo, vec1, vec2); | 
					
						
							| 
									
										
										
										
											2009-06-03 23:56:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 				//is->dist = (u1+dda_lambda*(u2-u1))*o_lambda;
 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				if (testnode(oc, is, no, ocval) ) | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 					found = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 				if (is->dist < (u1 + dda_lambda * (u2 - u1)) * o_lambda) | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 					return found; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-11 16:44:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			lambda_o = dda_lambda; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-11-12 07:33:01 +00:00
										 |  |  | 			/* traversing octree nodes need careful detection of smallest values, with proper
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			 * exceptions for equal lambdas */ | 
					
						
							|  |  |  | 			eqval = (lambda_x == lambda_y); | 
					
						
							|  |  |  | 			if (lambda_y == lambda_z) eqval += 2; | 
					
						
							|  |  |  | 			if (lambda_x == lambda_z) eqval += 4; | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (eqval) {    // only 4 cases exist!
 | 
					
						
							|  |  |  | 				if (eqval == 7) { // x=y=z
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					xo += dx; lambda_x += ldx; | 
					
						
							|  |  |  | 					yo += dy; lambda_y += ldy; | 
					
						
							|  |  |  | 					zo += dz; lambda_z += ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				else if (eqval == 1) { // x=y
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					if (lambda_y < lambda_z) { | 
					
						
							|  |  |  | 						xo += dx; lambda_x += ldx; | 
					
						
							|  |  |  | 						yo += dy; lambda_y += ldy; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 						zo += dz; lambda_z += ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				else if (eqval == 2) { // y=z
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					if (lambda_x < lambda_y) { | 
					
						
							|  |  |  | 						xo += dx; lambda_x += ldx; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 						yo += dy; lambda_y += ldy; | 
					
						
							|  |  |  | 						zo += dz; lambda_z += ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { // x=z
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					if (lambda_y < lambda_x) { | 
					
						
							|  |  |  | 						yo += dy; lambda_y += ldy; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 						xo += dx; lambda_x += ldx; | 
					
						
							|  |  |  | 						zo += dz; lambda_z += ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 			else {  // all three different, just three cases exist
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 				eqval = (lambda_x < lambda_y); | 
					
						
							|  |  |  | 				if (lambda_y < lambda_z) eqval += 2; | 
					
						
							|  |  |  | 				if (lambda_x < lambda_z) eqval += 4; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				if (eqval == 7 || eqval == 5) { // x smallest
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					xo += dx; lambda_x += ldx; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-06-05 21:54:21 +00:00
										 |  |  | 				else if (eqval == 2 || eqval == 6) { // y smallest
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					yo += dy; lambda_y += ldy; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { // z smallest
 | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 					zo += dz; lambda_z += ldz; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-21 05:07:26 +00:00
										 |  |  | 			dda_lambda = min_fff(lambda_x, lambda_y, lambda_z); | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			if (dda_lambda == lambda_o) break; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 			/* to make sure the last node is always checked */ | 
					
						
							| 
									
										
										
										
											2012-12-11 14:18:37 +00:00
										 |  |  | 			if (lambda_o >= 1.0f) break; | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* reached end, no intersections found */ | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | }	 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-10 21:02:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-07-26 13:38:24 +00:00
										 |  |  | 
 |