| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2013-08-18 14:16:15 +00:00
										 |  |  |  * Copyright 2011-2013 Blender Foundation | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-08-18 14:16:15 +00:00
										 |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * You may obtain a copy of the License at | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-08-18 14:16:15 +00:00
										 |  |  |  * http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-08-18 14:16:15 +00:00
										 |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							| 
									
										
										
										
											2014-12-25 02:50:24 +01:00
										 |  |  |  * limitations under the License. | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Cycles: Make all #include statements relative to cycles source directory
The idea is to make include statements more explicit and obvious where the
file is coming from, additionally reducing chance of wrong header being
picked up.
For example, it was not obvious whether bvh.h was refferring to builder
or traversal, whenter node.h is a generic graph node or a shader node
and cases like that.
Surely this might look obvious for the active developers, but after some
time of not touching the code it becomes less obvious where file is coming
from.
This was briefly mentioned in T50824 and seems @brecht is fine with such
explicitness, but need to agree with all active developers before committing
this.
Please note that this patch is lacking changes related on GPU/OpenCL
support. This will be solved if/when we all agree this is a good idea to move
forward.
Reviewers: brecht, lukasstockner97, maiself, nirved, dingto, juicyfruit, swerner
Reviewed By: lukasstockner97, maiself, nirved, dingto
Subscribers: brecht
Differential Revision: https://developer.blender.org/D2586
											
										 
											2017-03-28 20:39:14 +02:00
										 |  |  | #include "render/camera.h"
 | 
					
						
							|  |  |  | #include "device/device.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  | #include "render/hair.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  | #include "render/integrator.h"
 | 
					
						
							| 
									
										
											  
											
												Cycles: Make all #include statements relative to cycles source directory
The idea is to make include statements more explicit and obvious where the
file is coming from, additionally reducing chance of wrong header being
picked up.
For example, it was not obvious whether bvh.h was refferring to builder
or traversal, whenter node.h is a generic graph node or a shader node
and cases like that.
Surely this might look obvious for the active developers, but after some
time of not touching the code it becomes less obvious where file is coming
from.
This was briefly mentioned in T50824 and seems @brecht is fine with such
explicitness, but need to agree with all active developers before committing
this.
Please note that this patch is lacking changes related on GPU/OpenCL
support. This will be solved if/when we all agree this is a good idea to move
forward.
Reviewers: brecht, lukasstockner97, maiself, nirved, dingto, juicyfruit, swerner
Reviewed By: lukasstockner97, maiself, nirved, dingto
Subscribers: brecht
Differential Revision: https://developer.blender.org/D2586
											
										 
											2017-03-28 20:39:14 +02:00
										 |  |  | #include "render/light.h"
 | 
					
						
							|  |  |  | #include "render/mesh.h"
 | 
					
						
							|  |  |  | #include "render/curves.h"
 | 
					
						
							|  |  |  | #include "render/object.h"
 | 
					
						
							|  |  |  | #include "render/particles.h"
 | 
					
						
							|  |  |  | #include "render/scene.h"
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Cycles: Make all #include statements relative to cycles source directory
The idea is to make include statements more explicit and obvious where the
file is coming from, additionally reducing chance of wrong header being
picked up.
For example, it was not obvious whether bvh.h was refferring to builder
or traversal, whenter node.h is a generic graph node or a shader node
and cases like that.
Surely this might look obvious for the active developers, but after some
time of not touching the code it becomes less obvious where file is coming
from.
This was briefly mentioned in T50824 and seems @brecht is fine with such
explicitness, but need to agree with all active developers before committing
this.
Please note that this patch is lacking changes related on GPU/OpenCL
support. This will be solved if/when we all agree this is a good idea to move
forward.
Reviewers: brecht, lukasstockner97, maiself, nirved, dingto, juicyfruit, swerner
Reviewed By: lukasstockner97, maiself, nirved, dingto
Subscribers: brecht
Differential Revision: https://developer.blender.org/D2586
											
										 
											2017-03-28 20:39:14 +02:00
										 |  |  | #include "util/util_foreach.h"
 | 
					
						
							|  |  |  | #include "util/util_logging.h"
 | 
					
						
							|  |  |  | #include "util/util_map.h"
 | 
					
						
							|  |  |  | #include "util/util_progress.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-07 04:05:47 +01:00
										 |  |  | #include "util/util_set.h"
 | 
					
						
							| 
									
										
											  
											
												Cycles: Make all #include statements relative to cycles source directory
The idea is to make include statements more explicit and obvious where the
file is coming from, additionally reducing chance of wrong header being
picked up.
For example, it was not obvious whether bvh.h was refferring to builder
or traversal, whenter node.h is a generic graph node or a shader node
and cases like that.
Surely this might look obvious for the active developers, but after some
time of not touching the code it becomes less obvious where file is coming
from.
This was briefly mentioned in T50824 and seems @brecht is fine with such
explicitness, but need to agree with all active developers before committing
this.
Please note that this patch is lacking changes related on GPU/OpenCL
support. This will be solved if/when we all agree this is a good idea to move
forward.
Reviewers: brecht, lukasstockner97, maiself, nirved, dingto, juicyfruit, swerner
Reviewed By: lukasstockner97, maiself, nirved, dingto
Subscribers: brecht
Differential Revision: https://developer.blender.org/D2586
											
										 
											2017-03-28 20:39:14 +02:00
										 |  |  | #include "util/util_vector.h"
 | 
					
						
							| 
									
										
										
										
											2018-10-28 05:37:41 -04:00
										 |  |  | #include "util/util_murmurhash.h"
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Cycles: Make all #include statements relative to cycles source directory
The idea is to make include statements more explicit and obvious where the
file is coming from, additionally reducing chance of wrong header being
picked up.
For example, it was not obvious whether bvh.h was refferring to builder
or traversal, whenter node.h is a generic graph node or a shader node
and cases like that.
Surely this might look obvious for the active developers, but after some
time of not touching the code it becomes less obvious where file is coming
from.
This was briefly mentioned in T50824 and seems @brecht is fine with such
explicitness, but need to agree with all active developers before committing
this.
Please note that this patch is lacking changes related on GPU/OpenCL
support. This will be solved if/when we all agree this is a good idea to move
forward.
Reviewers: brecht, lukasstockner97, maiself, nirved, dingto, juicyfruit, swerner
Reviewed By: lukasstockner97, maiself, nirved, dingto
Subscribers: brecht
Differential Revision: https://developer.blender.org/D2586
											
										 
											2017-03-28 20:39:14 +02:00
										 |  |  | #include "subd/subd_patch_table.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | CCL_NAMESPACE_BEGIN | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | /* Global state of object transform update. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct UpdateObjectTransformState { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Global state used by device_update_object_transform().
 | 
					
						
							|  |  |  |    * Common for both threaded and non-threaded update. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Type of the motion required by the scene settings. */ | 
					
						
							|  |  |  |   Scene::MotionType need_motion; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Mapping from particle system to a index in packed particle array.
 | 
					
						
							|  |  |  |    * Only used for read. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   map<ParticleSystem *, int> particle_offset; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Mesh area.
 | 
					
						
							|  |  |  |    * Used to avoid calculation of mesh area multiple times. Used for both | 
					
						
							|  |  |  |    * read and write. Acquire surface_area_lock to keep it all thread safe. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   map<Mesh *, float> surface_area_map; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Motion offsets for each object. */ | 
					
						
							|  |  |  |   array<uint> motion_offset; | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Packed object arrays. Those will be filled in. */ | 
					
						
							|  |  |  |   uint *object_flag; | 
					
						
							|  |  |  |   KernelObject *objects; | 
					
						
							|  |  |  |   Transform *object_motion_pass; | 
					
						
							|  |  |  |   DecomposedTransform *object_motion; | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |   float *object_volume_step; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Flags which will be synchronized to Integrator. */ | 
					
						
							|  |  |  |   bool have_motion; | 
					
						
							|  |  |  |   bool have_curves; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* ** Scheduling queue. ** */ | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Scene *scene; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Some locks to keep everything thread-safe. */ | 
					
						
							|  |  |  |   thread_spin_lock queue_lock; | 
					
						
							|  |  |  |   thread_spin_lock surface_area_lock; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* First unused object index in the queue. */ | 
					
						
							|  |  |  |   int queue_start_object; | 
					
						
							| 
									
										
										
										
											2018-03-10 01:15:02 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | /* Object */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-07 21:44:17 +02:00
										 |  |  | NODE_DEFINE(Object) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   NodeType *type = NodeType::add("object", create); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   SOCKET_NODE(geometry, "Geometry", &Geometry::node_base_type); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   SOCKET_TRANSFORM(tfm, "Transform", transform_identity()); | 
					
						
							|  |  |  |   SOCKET_UINT(visibility, "Visibility", ~0); | 
					
						
							| 
									
										
										
										
											2019-08-22 14:26:09 +02:00
										 |  |  |   SOCKET_COLOR(color, "Color", make_float3(0.0f, 0.0f, 0.0f)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   SOCKET_UINT(random_id, "Random ID", 0); | 
					
						
							|  |  |  |   SOCKET_INT(pass_id, "Pass ID", 0); | 
					
						
							|  |  |  |   SOCKET_BOOLEAN(use_holdout, "Use Holdout", false); | 
					
						
							|  |  |  |   SOCKET_BOOLEAN(hide_on_missing_motion, "Hide on Missing Motion", false); | 
					
						
							|  |  |  |   SOCKET_POINT(dupli_generated, "Dupli Generated", make_float3(0.0f, 0.0f, 0.0f)); | 
					
						
							|  |  |  |   SOCKET_POINT2(dupli_uv, "Dupli UV", make_float2(0.0f, 0.0f)); | 
					
						
							|  |  |  |   SOCKET_TRANSFORM_ARRAY(motion, "Motion", array<Transform>()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   SOCKET_BOOLEAN(is_shadow_catcher, "Shadow Catcher", false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return type; | 
					
						
							| 
									
										
										
										
											2016-05-07 21:44:17 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | Object::Object() : Node(node_type) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   particle_system = NULL; | 
					
						
							|  |  |  |   particle_index = 0; | 
					
						
							|  |  |  |   bounds = BoundBox::empty; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Object::~Object() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | void Object::update_motion() | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!use_motion()) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool have_motion = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (size_t i = 0; i < motion.size(); i++) { | 
					
						
							|  |  |  |     if (motion[i] == transform_empty()) { | 
					
						
							|  |  |  |       if (hide_on_missing_motion) { | 
					
						
							|  |  |  |         /* Hide objects that have no valid previous or next
 | 
					
						
							|  |  |  |          * transform, for example particle that stop existing. It | 
					
						
							|  |  |  |          * would be better to handle this in the kernel and make | 
					
						
							|  |  |  |          * objects invisible outside certain motion steps. */ | 
					
						
							|  |  |  |         tfm = transform_empty(); | 
					
						
							|  |  |  |         motion.clear(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         /* Otherwise just copy center motion. */ | 
					
						
							|  |  |  |         motion[i] = tfm; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Test if any of the transforms are actually different. */ | 
					
						
							|  |  |  |     have_motion = have_motion || motion[i] != tfm; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Clear motion array if there is no actual motion. */ | 
					
						
							|  |  |  |   if (!have_motion) { | 
					
						
							|  |  |  |     motion.clear(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Object::compute_bounds(bool motion_blur) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   BoundBox mbounds = geometry->bounds; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (motion_blur && use_motion()) { | 
					
						
							|  |  |  |     array<DecomposedTransform> decomp(motion.size()); | 
					
						
							|  |  |  |     transform_motion_decompose(decomp.data(), motion.data(), motion.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bounds = BoundBox::empty; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* todo: this is really terrible. according to pbrt there is a better
 | 
					
						
							|  |  |  |      * way to find this iteratively, but did not find implementation yet | 
					
						
							|  |  |  |      * or try to implement myself */ | 
					
						
							|  |  |  |     for (float t = 0.0f; t < 1.0f; t += (1.0f / 128.0f)) { | 
					
						
							|  |  |  |       Transform ttfm; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       transform_motion_array_interpolate(&ttfm, decomp.data(), motion.size(), t); | 
					
						
							|  |  |  |       bounds.grow(mbounds.transformed(&ttfm)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* No motion blur case. */ | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     if (geometry->transform_applied) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       bounds = mbounds; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       bounds = mbounds.transformed(&tfm); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-09 17:02:08 +02:00
										 |  |  | void Object::apply_transform(bool apply_to_motion) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   if (!geometry || tfm == transform_identity()) | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   geometry->apply_transform(tfm, apply_to_motion); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* we keep normals pointing in same direction on negative scale, notify
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |    * geometry about this in it (re)calculates normals */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (transform_negative_scale(tfm)) | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     geometry->transform_negative_scaled = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (bounds.valid()) { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     geometry->compute_bounds(); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     compute_bounds(false); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* tfm is not reset to identity, all code that uses it needs to check the
 | 
					
						
							|  |  |  |    * transform_applied boolean */ | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Object::tag_update(Scene *scene) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   if (geometry) { | 
					
						
							|  |  |  |     if (geometry->transform_applied) | 
					
						
							|  |  |  |       geometry->need_update = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     foreach (Shader *shader, geometry->used_shaders) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (shader->use_mis && shader->has_surface_emission) | 
					
						
							|  |  |  |         scene->light_manager->need_update = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   scene->camera->need_flags_update = true; | 
					
						
							|  |  |  |   scene->curve_system_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   scene->geometry_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   scene->object_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | bool Object::use_motion() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return (motion.size() > 1); | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | float Object::motion_time(int step) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return (use_motion()) ? 2.0f * step / (motion.size() - 1) - 1.0f : 0.0f; | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int Object::motion_step(float time) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (use_motion()) { | 
					
						
							|  |  |  |     for (size_t step = 0; step < motion.size(); step++) { | 
					
						
							|  |  |  |       if (time == motion_time(step)) { | 
					
						
							|  |  |  |         return step; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return -1; | 
					
						
							| 
									
										
										
										
											2018-03-08 04:04:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-10 00:37:07 +01:00
										 |  |  | bool Object::is_traceable() const | 
					
						
							| 
									
										
										
										
											2016-06-06 09:15:34 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Mesh itself can be empty,can skip all such objects. */ | 
					
						
							|  |  |  |   if (!bounds.valid() || bounds.size() == make_float3(0.0f, 0.0f, 0.0f)) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   /* TODO(sergey): Check for mesh vertices/curves. visibility flags. */ | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2016-06-06 09:15:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | uint Object::visibility_for_tracing() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint trace_visibility = visibility; | 
					
						
							|  |  |  |   if (is_shadow_catcher) { | 
					
						
							|  |  |  |     trace_visibility &= ~PATH_RAY_SHADOW_NON_CATCHER; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     trace_visibility &= ~PATH_RAY_SHADOW_CATCHER; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return trace_visibility; | 
					
						
							| 
									
										
										
										
											2017-08-10 09:19:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  | float Object::compute_volume_step_size() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!geometry->has_volume) { | 
					
						
							|  |  |  |     return FLT_MAX; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Compute step rate from shaders. */ | 
					
						
							|  |  |  |   float step_rate = FLT_MAX; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   foreach (Shader *shader, geometry->used_shaders) { | 
					
						
							|  |  |  |     if (shader->has_volume) { | 
					
						
							|  |  |  |       if ((shader->heterogeneous_volume && shader->has_volume_spatial_varying) || | 
					
						
							|  |  |  |           (shader->has_volume_attribute_dependency)) { | 
					
						
							|  |  |  |         step_rate = fminf(shader->volume_step_rate, step_rate); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (step_rate == FLT_MAX) { | 
					
						
							|  |  |  |     return FLT_MAX; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Compute step size from voxel grids. */ | 
					
						
							|  |  |  |   float step_size = FLT_MAX; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   foreach (Attribute &attr, geometry->attributes.attributes) { | 
					
						
							|  |  |  |     if (attr.element == ATTR_ELEMENT_VOXEL) { | 
					
						
							|  |  |  |       ImageHandle &handle = attr.data_voxel(); | 
					
						
							|  |  |  |       const ImageMetaData &metadata = handle.metadata(); | 
					
						
							|  |  |  |       if (metadata.width == 0 || metadata.height == 0 || metadata.depth == 0) { | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Step size is transformed from voxel to world space. */ | 
					
						
							|  |  |  |       Transform voxel_tfm = tfm; | 
					
						
							|  |  |  |       if (metadata.use_transform_3d) { | 
					
						
							|  |  |  |         voxel_tfm = tfm * transform_inverse(metadata.transform_3d); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       float3 size = make_float3( | 
					
						
							|  |  |  |           1.0f / metadata.width, 1.0f / metadata.height, 1.0f / metadata.depth); | 
					
						
							|  |  |  |       float voxel_step_size = min3(fabs(transform_direction(&voxel_tfm, size))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (voxel_step_size > 0.0f) { | 
					
						
							|  |  |  |         step_size = fminf(voxel_step_size, step_size); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (step_size == FLT_MAX) { | 
					
						
							|  |  |  |     /* Fall back to 1/10th of bounds for procedural volumes. */ | 
					
						
							|  |  |  |     step_size = 0.1f * average(bounds.size()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   step_size *= step_rate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return step_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-29 02:06:30 +01:00
										 |  |  | int Object::get_device_index() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return index; | 
					
						
							| 
									
										
										
										
											2018-11-29 02:06:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | /* Object Manager */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ObjectManager::ObjectManager() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   need_update = true; | 
					
						
							|  |  |  |   need_flags_update = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ObjectManager::~ObjectManager() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  | static float object_surface_area(UpdateObjectTransformState *state, | 
					
						
							|  |  |  |                                  const Transform &tfm, | 
					
						
							|  |  |  |                                  Geometry *geom) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   if (geom->type != Geometry::MESH) { | 
					
						
							|  |  |  |     return 0.0f; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Compute surface area. for uniform scale we can do avoid the many
 | 
					
						
							|  |  |  |    * transform calls and share computation for instances. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * TODO(brecht): Correct for displacement, and move to a better place. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   Mesh *mesh = static_cast<Mesh *>(geom); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float surface_area = 0.0f; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   float uniform_scale; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (transform_uniform_scale(tfm, uniform_scale)) { | 
					
						
							|  |  |  |     map<Mesh *, float>::iterator it; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* NOTE: This isn't fully optimal and could in theory lead to multiple
 | 
					
						
							|  |  |  |      * threads calculating area of the same mesh in parallel. However, this | 
					
						
							|  |  |  |      * also prevents suspending all the threads when some mesh's area is | 
					
						
							|  |  |  |      * not yet known. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     state->surface_area_lock.lock(); | 
					
						
							|  |  |  |     it = state->surface_area_map.find(mesh); | 
					
						
							|  |  |  |     state->surface_area_lock.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (it == state->surface_area_map.end()) { | 
					
						
							|  |  |  |       size_t num_triangles = mesh->num_triangles(); | 
					
						
							|  |  |  |       for (size_t j = 0; j < num_triangles; j++) { | 
					
						
							|  |  |  |         Mesh::Triangle t = mesh->get_triangle(j); | 
					
						
							|  |  |  |         float3 p1 = mesh->verts[t.v[0]]; | 
					
						
							|  |  |  |         float3 p2 = mesh->verts[t.v[1]]; | 
					
						
							|  |  |  |         float3 p3 = mesh->verts[t.v[2]]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         surface_area += triangle_area(p1, p2, p3); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       state->surface_area_lock.lock(); | 
					
						
							|  |  |  |       state->surface_area_map[mesh] = surface_area; | 
					
						
							|  |  |  |       state->surface_area_lock.unlock(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       surface_area = it->second; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     surface_area *= uniform_scale; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     size_t num_triangles = mesh->num_triangles(); | 
					
						
							|  |  |  |     for (size_t j = 0; j < num_triangles; j++) { | 
					
						
							|  |  |  |       Mesh::Triangle t = mesh->get_triangle(j); | 
					
						
							|  |  |  |       float3 p1 = transform_point(&tfm, mesh->verts[t.v[0]]); | 
					
						
							|  |  |  |       float3 p2 = transform_point(&tfm, mesh->verts[t.v[1]]); | 
					
						
							|  |  |  |       float3 p3 = transform_point(&tfm, mesh->verts[t.v[2]]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       surface_area += triangle_area(p1, p2, p3); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   return surface_area; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ObjectManager::device_update_object_transform(UpdateObjectTransformState *state, Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   KernelObject &kobject = state->objects[ob->index]; | 
					
						
							|  |  |  |   Transform *object_motion_pass = state->object_motion_pass; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Geometry *geom = ob->geometry; | 
					
						
							|  |  |  |   uint flag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Compute transformations. */ | 
					
						
							|  |  |  |   Transform tfm = ob->tfm; | 
					
						
							|  |  |  |   Transform itfm = transform_inverse(tfm); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   float3 color = ob->color; | 
					
						
							|  |  |  |   float pass_id = ob->pass_id; | 
					
						
							|  |  |  |   float random_number = (float)ob->random_id * (1.0f / (float)0xFFFFFFFF); | 
					
						
							|  |  |  |   int particle_index = (ob->particle_system) ? | 
					
						
							|  |  |  |                            ob->particle_index + state->particle_offset[ob->particle_system] : | 
					
						
							|  |  |  |                            0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   kobject.tfm = tfm; | 
					
						
							|  |  |  |   kobject.itfm = itfm; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   kobject.surface_area = object_surface_area(state, tfm, geom); | 
					
						
							| 
									
										
										
										
											2019-08-22 14:26:09 +02:00
										 |  |  |   kobject.color[0] = color.x; | 
					
						
							|  |  |  |   kobject.color[1] = color.y; | 
					
						
							|  |  |  |   kobject.color[2] = color.z; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   kobject.pass_id = pass_id; | 
					
						
							|  |  |  |   kobject.random_number = random_number; | 
					
						
							|  |  |  |   kobject.particle_index = particle_index; | 
					
						
							|  |  |  |   kobject.motion_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   if (geom->use_motion_blur) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     state->have_motion = true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (geom->type == Geometry::MESH) { | 
					
						
							|  |  |  |     /* TODO: why only mesh? */ | 
					
						
							|  |  |  |     Mesh *mesh = static_cast<Mesh *>(geom); | 
					
						
							|  |  |  |     if (mesh->attributes.find(ATTR_STD_MOTION_VERTEX_POSITION)) { | 
					
						
							|  |  |  |       flag |= SD_OBJECT_HAS_VERTEX_MOTION; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (state->need_motion == Scene::MOTION_PASS) { | 
					
						
							|  |  |  |     /* Clear motion array if there is no actual motion. */ | 
					
						
							|  |  |  |     ob->update_motion(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Compute motion transforms. */ | 
					
						
							|  |  |  |     Transform tfm_pre, tfm_post; | 
					
						
							|  |  |  |     if (ob->use_motion()) { | 
					
						
							|  |  |  |       tfm_pre = ob->motion[0]; | 
					
						
							|  |  |  |       tfm_post = ob->motion[ob->motion.size() - 1]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       tfm_pre = tfm; | 
					
						
							|  |  |  |       tfm_post = tfm; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Motion transformations, is world/object space depending if mesh
 | 
					
						
							|  |  |  |      * comes with deformed position in object space, or if we transform | 
					
						
							|  |  |  |      * the shading point in world space. */ | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     if (!(flag & SD_OBJECT_HAS_VERTEX_MOTION)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       tfm_pre = tfm_pre * itfm; | 
					
						
							|  |  |  |       tfm_post = tfm_post * itfm; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int motion_pass_offset = ob->index * OBJECT_MOTION_PASS_SIZE; | 
					
						
							|  |  |  |     object_motion_pass[motion_pass_offset + 0] = tfm_pre; | 
					
						
							|  |  |  |     object_motion_pass[motion_pass_offset + 1] = tfm_post; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (state->need_motion == Scene::MOTION_BLUR) { | 
					
						
							|  |  |  |     if (ob->use_motion()) { | 
					
						
							|  |  |  |       kobject.motion_offset = state->motion_offset[ob->index]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Decompose transforms for interpolation. */ | 
					
						
							|  |  |  |       DecomposedTransform *decomp = state->object_motion + kobject.motion_offset; | 
					
						
							|  |  |  |       transform_motion_decompose(decomp, ob->motion.data(), ob->motion.size()); | 
					
						
							|  |  |  |       flag |= SD_OBJECT_MOTION; | 
					
						
							|  |  |  |       state->have_motion = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Dupli object coords and motion info. */ | 
					
						
							|  |  |  |   kobject.dupli_generated[0] = ob->dupli_generated[0]; | 
					
						
							|  |  |  |   kobject.dupli_generated[1] = ob->dupli_generated[1]; | 
					
						
							|  |  |  |   kobject.dupli_generated[2] = ob->dupli_generated[2]; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   kobject.numkeys = (geom->type == Geometry::HAIR) ? static_cast<Hair *>(geom)->curve_keys.size() : | 
					
						
							|  |  |  |                                                      0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   kobject.dupli_uv[0] = ob->dupli_uv[0]; | 
					
						
							|  |  |  |   kobject.dupli_uv[1] = ob->dupli_uv[1]; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   int totalsteps = geom->motion_steps; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   kobject.numsteps = (totalsteps - 1) / 2; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   kobject.numverts = (geom->type == Geometry::MESH) ? static_cast<Mesh *>(geom)->verts.size() : 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   kobject.patch_map_offset = 0; | 
					
						
							|  |  |  |   kobject.attribute_map_offset = 0; | 
					
						
							|  |  |  |   uint32_t hash_name = util_murmur_hash3(ob->name.c_str(), ob->name.length(), 0); | 
					
						
							|  |  |  |   uint32_t hash_asset = util_murmur_hash3(ob->asset_name.c_str(), ob->asset_name.length(), 0); | 
					
						
							|  |  |  |   kobject.cryptomatte_object = util_hash_to_float(hash_name); | 
					
						
							|  |  |  |   kobject.cryptomatte_asset = util_hash_to_float(hash_asset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Object flag. */ | 
					
						
							|  |  |  |   if (ob->use_holdout) { | 
					
						
							|  |  |  |     flag |= SD_OBJECT_HOLDOUT_MASK; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   state->object_flag[ob->index] = flag; | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |   state->object_volume_step[ob->index] = FLT_MAX; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Have curves. */ | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   if (geom->type == Geometry::HAIR) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     state->have_curves = true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:26 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-04 21:40:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | bool ObjectManager::device_update_object_transform_pop_work(UpdateObjectTransformState *state, | 
					
						
							|  |  |  |                                                             int *start_index, | 
					
						
							|  |  |  |                                                             int *num_objects) | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:26 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Tweakable parameter, number of objects per chunk.
 | 
					
						
							|  |  |  |    * Too small value will cause some extra overhead due to spin lock, | 
					
						
							|  |  |  |    * too big value might not use all threads nicely. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static const int OBJECTS_PER_TASK = 32; | 
					
						
							|  |  |  |   bool have_work = false; | 
					
						
							|  |  |  |   state->queue_lock.lock(); | 
					
						
							|  |  |  |   int num_scene_objects = state->scene->objects.size(); | 
					
						
							|  |  |  |   if (state->queue_start_object < num_scene_objects) { | 
					
						
							|  |  |  |     int count = min(OBJECTS_PER_TASK, num_scene_objects - state->queue_start_object); | 
					
						
							|  |  |  |     *start_index = state->queue_start_object; | 
					
						
							|  |  |  |     *num_objects = count; | 
					
						
							|  |  |  |     state->queue_start_object += count; | 
					
						
							|  |  |  |     have_work = true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   state->queue_lock.unlock(); | 
					
						
							|  |  |  |   return have_work; | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:26 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-05-02 09:33:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void ObjectManager::device_update_object_transform_task(UpdateObjectTransformState *state) | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:26 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int start_index, num_objects; | 
					
						
							|  |  |  |   while (device_update_object_transform_pop_work(state, &start_index, &num_objects)) { | 
					
						
							|  |  |  |     for (int i = 0; i < num_objects; ++i) { | 
					
						
							|  |  |  |       const int object_index = start_index + i; | 
					
						
							|  |  |  |       Object *ob = state->scene->objects[object_index]; | 
					
						
							|  |  |  |       device_update_object_transform(state, ob); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:26 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-04-17 20:07:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void ObjectManager::device_update_transforms(DeviceScene *dscene, Scene *scene, Progress &progress) | 
					
						
							| 
									
										
										
										
											2016-04-20 18:12:26 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   UpdateObjectTransformState state; | 
					
						
							|  |  |  |   state.need_motion = scene->need_motion(); | 
					
						
							|  |  |  |   state.have_motion = false; | 
					
						
							|  |  |  |   state.have_curves = false; | 
					
						
							|  |  |  |   state.scene = scene; | 
					
						
							|  |  |  |   state.queue_start_object = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   state.objects = dscene->objects.alloc(scene->objects.size()); | 
					
						
							|  |  |  |   state.object_flag = dscene->object_flag.alloc(scene->objects.size()); | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |   state.object_volume_step = dscene->object_volume_step.alloc(scene->objects.size()); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   state.object_motion = NULL; | 
					
						
							|  |  |  |   state.object_motion_pass = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (state.need_motion == Scene::MOTION_PASS) { | 
					
						
							|  |  |  |     state.object_motion_pass = dscene->object_motion_pass.alloc(OBJECT_MOTION_PASS_SIZE * | 
					
						
							|  |  |  |                                                                 scene->objects.size()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (state.need_motion == Scene::MOTION_BLUR) { | 
					
						
							|  |  |  |     /* Set object offsets into global object motion array. */ | 
					
						
							|  |  |  |     uint *motion_offsets = state.motion_offset.resize(scene->objects.size()); | 
					
						
							|  |  |  |     uint motion_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     foreach (Object *ob, scene->objects) { | 
					
						
							|  |  |  |       *motion_offsets = motion_offset; | 
					
						
							|  |  |  |       motion_offsets++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Clear motion array if there is no actual motion. */ | 
					
						
							|  |  |  |       ob->update_motion(); | 
					
						
							|  |  |  |       motion_offset += ob->motion.size(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     state.object_motion = dscene->object_motion.alloc(motion_offset); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Particle system device offsets
 | 
					
						
							|  |  |  |    * 0 is dummy particle, index starts at 1. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   int numparticles = 1; | 
					
						
							|  |  |  |   foreach (ParticleSystem *psys, scene->particle_systems) { | 
					
						
							|  |  |  |     state.particle_offset[psys] = numparticles; | 
					
						
							|  |  |  |     numparticles += psys->particles.size(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* NOTE: If it's just a handful of objects we deal with them in a single
 | 
					
						
							|  |  |  |    * thread to avoid threading overhead. However, this threshold is might | 
					
						
							|  |  |  |    * need some tweaks to make mid-complex scenes optimal. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   if (scene->objects.size() < 64) { | 
					
						
							|  |  |  |     foreach (Object *ob, scene->objects) { | 
					
						
							|  |  |  |       device_update_object_transform(&state, ob); | 
					
						
							|  |  |  |       if (progress.get_cancel()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     const int num_threads = TaskScheduler::num_threads(); | 
					
						
							|  |  |  |     TaskPool pool; | 
					
						
							|  |  |  |     for (int i = 0; i < num_threads; ++i) { | 
					
						
							|  |  |  |       pool.push(function_bind(&ObjectManager::device_update_object_transform_task, this, &state)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pool.wait_work(); | 
					
						
							|  |  |  |     if (progress.get_cancel()) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dscene->objects.copy_to_device(); | 
					
						
							|  |  |  |   if (state.need_motion == Scene::MOTION_PASS) { | 
					
						
							|  |  |  |     dscene->object_motion_pass.copy_to_device(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (state.need_motion == Scene::MOTION_BLUR) { | 
					
						
							|  |  |  |     dscene->object_motion.copy_to_device(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dscene->data.bvh.have_motion = state.have_motion; | 
					
						
							|  |  |  |   dscene->data.bvh.have_curves = state.have_curves; | 
					
						
							|  |  |  |   dscene->data.bvh.have_instancing = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void ObjectManager::device_update(Device *device, | 
					
						
							|  |  |  |                                   DeviceScene *dscene, | 
					
						
							|  |  |  |                                   Scene *scene, | 
					
						
							|  |  |  |                                   Progress &progress) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!need_update) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2016-04-22 10:55:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   VLOG(1) << "Total " << scene->objects.size() << " objects."; | 
					
						
							| 
									
										
										
										
											2016-04-22 10:55:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   device_free(device, dscene); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (scene->objects.size() == 0) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Assign object IDs. */ | 
					
						
							|  |  |  |   int index = 0; | 
					
						
							|  |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							|  |  |  |     object->index = index++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-29 02:06:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* set object transform matrices, before applying static transforms */ | 
					
						
							|  |  |  |   progress.set_status("Updating Objects", "Copying Transformations to device"); | 
					
						
							|  |  |  |   device_update_transforms(dscene, scene, progress); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (progress.get_cancel()) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* prepare for static BVH building */ | 
					
						
							|  |  |  |   /* todo: do before to support getting object level coords? */ | 
					
						
							|  |  |  |   if (scene->params.bvh_type == SceneParams::BVH_STATIC) { | 
					
						
							|  |  |  |     progress.set_status("Updating Objects", "Applying Static Transformations"); | 
					
						
							|  |  |  |     apply_static_transforms(dscene, scene, progress); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void ObjectManager::device_update_flags( | 
					
						
							|  |  |  |     Device *, DeviceScene *dscene, Scene *scene, Progress & /*progress*/, bool bounds_valid) | 
					
						
							| 
									
										
										
										
											2014-10-03 12:11:19 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (!need_update && !need_flags_update) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   need_update = false; | 
					
						
							|  |  |  |   need_flags_update = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (scene->objects.size() == 0) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Object info flag. */ | 
					
						
							|  |  |  |   uint *object_flag = dscene->object_flag.data(); | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |   float *object_volume_step = dscene->object_volume_step.data(); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Object volume intersection. */ | 
					
						
							|  |  |  |   vector<Object *> volume_objects; | 
					
						
							|  |  |  |   bool has_volume_objects = false; | 
					
						
							|  |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     if (object->geometry->has_volume) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (bounds_valid) { | 
					
						
							|  |  |  |         volume_objects.push_back(object); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       has_volume_objects = true; | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |       object_volume_step[object->index] = object->compute_volume_step_size(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       object_volume_step[object->index] = FLT_MAX; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     if (object->geometry->has_volume) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       object_flag[object->index] |= SD_OBJECT_HAS_VOLUME; | 
					
						
							|  |  |  |       object_flag[object->index] &= ~SD_OBJECT_HAS_VOLUME_ATTRIBUTES; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |       foreach (Attribute &attr, object->geometry->attributes.attributes) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         if (attr.element == ATTR_ELEMENT_VOXEL) { | 
					
						
							|  |  |  |           object_flag[object->index] |= SD_OBJECT_HAS_VOLUME_ATTRIBUTES; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       object_flag[object->index] &= ~(SD_OBJECT_HAS_VOLUME | SD_OBJECT_HAS_VOLUME_ATTRIBUTES); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (object->is_shadow_catcher) { | 
					
						
							|  |  |  |       object_flag[object->index] |= SD_OBJECT_SHADOW_CATCHER; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       object_flag[object->index] &= ~SD_OBJECT_SHADOW_CATCHER; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (bounds_valid) { | 
					
						
							|  |  |  |       foreach (Object *volume_object, volume_objects) { | 
					
						
							|  |  |  |         if (object == volume_object) { | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (object->bounds.intersects(volume_object->bounds)) { | 
					
						
							|  |  |  |           object_flag[object->index] |= SD_OBJECT_INTERSECTS_VOLUME; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (has_volume_objects) { | 
					
						
							|  |  |  |       /* Not really valid, but can't make more reliable in the case
 | 
					
						
							|  |  |  |        * of bounds not being up to date. | 
					
						
							|  |  |  |        */ | 
					
						
							|  |  |  |       object_flag[object->index] |= SD_OBJECT_INTERSECTS_VOLUME; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Copy object flag. */ | 
					
						
							|  |  |  |   dscene->object_flag.copy_to_device(); | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |   dscene->object_volume_step.copy_to_device(); | 
					
						
							| 
									
										
										
										
											2014-10-03 12:11:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 17:40:36 +01:00
										 |  |  | void ObjectManager::device_update_mesh_offsets(Device *, DeviceScene *dscene, Scene *scene) | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (dscene->objects.size() == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-08-09 04:32:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   KernelObject *kobjects = dscene->objects.data(); | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool update = false; | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     Geometry *geom = object->geometry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (geom->type == Geometry::MESH) { | 
					
						
							|  |  |  |       Mesh *mesh = static_cast<Mesh *>(geom); | 
					
						
							|  |  |  |       if (mesh->patch_table) { | 
					
						
							|  |  |  |         uint patch_map_offset = 2 * (mesh->patch_table_offset + mesh->patch_table->total_size() - | 
					
						
							|  |  |  |                                      mesh->patch_table->num_nodes * PATCH_NODE_SIZE) - | 
					
						
							|  |  |  |                                 mesh->patch_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (kobjects[object->index].patch_map_offset != patch_map_offset) { | 
					
						
							|  |  |  |           kobjects[object->index].patch_map_offset = patch_map_offset; | 
					
						
							|  |  |  |           update = true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     if (kobjects[object->index].attribute_map_offset != geom->attr_map_offset) { | 
					
						
							|  |  |  |       kobjects[object->index].attribute_map_offset = geom->attr_map_offset; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       update = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (update) { | 
					
						
							|  |  |  |     dscene->objects.copy_to_device(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-07-16 22:57:06 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-21 01:09:59 +02:00
										 |  |  | void ObjectManager::device_free(Device *, DeviceScene *dscene) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   dscene->objects.free(); | 
					
						
							|  |  |  |   dscene->object_motion_pass.free(); | 
					
						
							|  |  |  |   dscene->object_motion.free(); | 
					
						
							|  |  |  |   dscene->object_flag.free(); | 
					
						
							| 
									
										
										
										
											2020-03-07 14:38:52 +01:00
										 |  |  |   dscene->object_volume_step.free(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void ObjectManager::apply_static_transforms(DeviceScene *dscene, Scene *scene, Progress &progress) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* todo: normals and displacement should be done before applying transform! */ | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   /* todo: create objects/geometry in right order! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   /* counter geometry users */ | 
					
						
							|  |  |  |   map<Geometry *, int> geometry_users; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Scene::MotionType need_motion = scene->need_motion(); | 
					
						
							|  |  |  |   bool motion_blur = need_motion == Scene::MOTION_BLUR; | 
					
						
							|  |  |  |   bool apply_to_motion = need_motion != Scene::MOTION_PASS; | 
					
						
							|  |  |  |   int i = 0; | 
					
						
							|  |  |  |   bool have_instancing = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     map<Geometry *, int>::iterator it = geometry_users.find(object->geometry); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     if (it == geometry_users.end()) | 
					
						
							|  |  |  |       geometry_users[object->geometry] = 1; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |       it->second++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (progress.get_cancel()) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint *object_flag = dscene->object_flag.data(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   /* apply transforms for objects with single user geometry */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							|  |  |  |     /* Annoying feedback loop here: we can't use is_instanced() because
 | 
					
						
							|  |  |  |      * it'll use uninitialized transform_applied flag. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |      * Could be solved by moving reference counter to Geometry. | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |     Geometry *geom = object->geometry; | 
					
						
							|  |  |  |     bool apply = (geometry_users[geom] == 1) && !geom->has_surface_bssrdf && | 
					
						
							|  |  |  |                  !geom->has_true_displacement(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (geom->type == Geometry::MESH) { | 
					
						
							|  |  |  |       Mesh *mesh = static_cast<Mesh *>(geom); | 
					
						
							|  |  |  |       apply = apply && mesh->subdivision_type == Mesh::SUBDIVISION_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (apply) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (!(motion_blur && object->use_motion())) { | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |         if (!geom->transform_applied) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           object->apply_transform(apply_to_motion); | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |           geom->transform_applied = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |           if (progress.get_cancel()) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         object_flag[i] |= SD_OBJECT_TRANSFORM_APPLIED; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |         if (geom->transform_negative_scaled) | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           object_flag[i] |= SD_OBJECT_NEGATIVE_SCALE_APPLIED; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         have_instancing = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       have_instancing = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     i++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dscene->data.bvh.have_instancing = have_instancing; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ObjectManager::tag_update(Scene *scene) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   need_update = true; | 
					
						
							|  |  |  |   scene->curve_system_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2020-02-02 12:04:19 +01:00
										 |  |  |   scene->geometry_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   scene->light_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-07 04:05:47 +01:00
										 |  |  | string ObjectManager::get_cryptomatte_objects(Scene *scene) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   string manifest = "{"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   unordered_set<ustring, ustringHash> objects; | 
					
						
							|  |  |  |   foreach (Object *object, scene->objects) { | 
					
						
							|  |  |  |     if (objects.count(object->name)) { | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     objects.insert(object->name); | 
					
						
							|  |  |  |     uint32_t hash_name = util_murmur_hash3(object->name.c_str(), object->name.length(), 0); | 
					
						
							|  |  |  |     manifest += string_printf("\"%s\":\"%08x\",", object->name.c_str(), hash_name); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   manifest[manifest.size() - 1] = '}'; | 
					
						
							|  |  |  |   return manifest; | 
					
						
							| 
									
										
										
										
											2018-11-07 04:05:47 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | string ObjectManager::get_cryptomatte_assets(Scene *scene) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   string manifest = "{"; | 
					
						
							|  |  |  |   unordered_set<ustring, ustringHash> assets; | 
					
						
							|  |  |  |   foreach (Object *ob, scene->objects) { | 
					
						
							|  |  |  |     if (assets.count(ob->asset_name)) { | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     assets.insert(ob->asset_name); | 
					
						
							|  |  |  |     uint32_t hash_asset = util_murmur_hash3(ob->asset_name.c_str(), ob->asset_name.length(), 0); | 
					
						
							|  |  |  |     manifest += string_printf("\"%s\":\"%08x\",", ob->asset_name.c_str(), hash_asset); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   manifest[manifest.size() - 1] = '}'; | 
					
						
							|  |  |  |   return manifest; | 
					
						
							| 
									
										
										
										
											2018-11-07 04:05:47 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | CCL_NAMESPACE_END |