| 
									
										
										
										
											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 "device/device.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 "render/graph.h"
 | 
					
						
							|  |  |  | #include "render/light.h"
 | 
					
						
							|  |  |  | #include "render/osl.h"
 | 
					
						
							|  |  |  | #include "render/scene.h"
 | 
					
						
							|  |  |  | #include "render/shader.h"
 | 
					
						
							|  |  |  | #include "render/nodes.h"
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef WITH_OSL
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												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 "kernel/osl/osl_globals.h"
 | 
					
						
							|  |  |  | #include "kernel/osl/osl_services.h"
 | 
					
						
							|  |  |  | #include "kernel/osl/osl_shader.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_md5.h"
 | 
					
						
							|  |  |  | #include "util/util_path.h"
 | 
					
						
							|  |  |  | #include "util/util_progress.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-08 06:48:14 +01:00
										 |  |  | #include "util/util_projection.h"
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CCL_NAMESPACE_BEGIN | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef WITH_OSL
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | /* Shared Texture and Shading System */ | 
					
						
							| 
									
										
										
										
											2013-04-04 23:48:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | OSL::TextureSystem *OSLShaderManager::ts_shared = NULL; | 
					
						
							|  |  |  | int OSLShaderManager::ts_shared_users = 0; | 
					
						
							|  |  |  | thread_mutex OSLShaderManager::ts_shared_mutex; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | OSL::ShadingSystem *OSLShaderManager::ss_shared = NULL; | 
					
						
							|  |  |  | OSLRenderServices *OSLShaderManager::services_shared = NULL; | 
					
						
							|  |  |  | int OSLShaderManager::ss_shared_users = 0; | 
					
						
							|  |  |  | thread_mutex OSLShaderManager::ss_shared_mutex; | 
					
						
							|  |  |  | thread_mutex OSLShaderManager::ss_mutex; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | /* Shader Manager */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OSLShaderManager::OSLShaderManager() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	texture_system_init(); | 
					
						
							| 
									
										
										
										
											2013-02-14 16:11:47 +00:00
										 |  |  | 	shading_system_init(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OSLShaderManager::~OSLShaderManager() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 	shading_system_free(); | 
					
						
							|  |  |  | 	texture_system_free(); | 
					
						
							| 
									
										
										
										
											2013-02-14 16:11:47 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 15:47:55 +05:00
										 |  |  | void OSLShaderManager::reset(Scene * /*scene*/) | 
					
						
							| 
									
										
										
										
											2013-02-14 16:11:47 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 	shading_system_free(); | 
					
						
							| 
									
										
										
										
											2013-02-14 16:11:47 +00:00
										 |  |  | 	shading_system_init(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLShaderManager::device_update(Device *device, DeviceScene *dscene, Scene *scene, Progress& progress) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(!need_update) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-22 10:55:26 +02:00
										 |  |  | 	VLOG(1) << "Total " << scene->shaders.size() << " shaders."; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 20:26:52 +00:00
										 |  |  | 	device_free(device, dscene, scene); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-30 11:51:17 +00:00
										 |  |  | 	/* determine which shaders are in use */ | 
					
						
							|  |  |  | 	device_update_shaders_used(scene); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	/* create shaders */ | 
					
						
							|  |  |  | 	OSLGlobals *og = (OSLGlobals*)device->osl_memory(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	foreach(Shader *shader, scene->shaders) { | 
					
						
							|  |  |  | 		assert(shader->graph); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(progress.get_cancel()) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 		/* we can only compile one shader at the time as the OSL ShadingSytem
 | 
					
						
							|  |  |  | 		 * has a single state, but we put the lock here so different renders can | 
					
						
							|  |  |  | 		 * compile shaders alternating */ | 
					
						
							|  |  |  | 		thread_scoped_lock lock(ss_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Cycles: Add Support for IES files as textures for light strength
This patch adds support for IES files, a file format that is commonly used to store the directional intensity distribution of light sources.
The new IES node is supposed to be plugged into the Strength input of the Emission node of the lamp.
Since people generating IES files do not really seem to care about the standard, the parser is flexible enough to accept all test files I have tried.
Some common weirdnesses are distributing values over multiple lines that should go into one line, using commas instead of spaces as delimiters and adding various useless stuff at the end of the file.
The user interface of the node is similar to the script node, the user can either select an internal Text or load a file.
Internally, IES files are handled similar to Image textures: They are stored in slots by the LightManager and each unique IES is assigned to one slot.
The local coordinate system of the lamp is used, so that the direction of the light can be changed. For UI reasons, it's usually best to add an area light,
rotate it and then change its type, since especially the point light does not immediately show its local coordinate system in the viewport.
Reviewers: #cycles, dingto, sergey, brecht
Reviewed By: #cycles, dingto, brecht
Subscribers: OgDEV, crazyrobinhood, secundar, cardboard, pisuke, intrah, swerner, micah_denn, harvester, gottfried, disnel, campbellbarton, duarteframos, Lapineige, brecht, juicyfruit, dingto, marek, rickyblender, bliblubli, lockal, sergey
Differential Revision: https://developer.blender.org/D1543
											
										 
											2018-05-27 00:46:37 +02:00
										 |  |  | 		OSLCompiler compiler((void*)this, (void*)ss, | 
					
						
							|  |  |  | 		                     scene->image_manager, | 
					
						
							|  |  |  | 		                     scene->light_manager); | 
					
						
							| 
									
										
										
										
											2016-05-14 14:50:03 +02:00
										 |  |  | 		compiler.background = (shader == scene->default_background); | 
					
						
							| 
									
										
										
										
											2015-11-20 18:18:27 +05:00
										 |  |  | 		compiler.compile(scene, og, shader); | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-18 09:36:00 +00:00
										 |  |  | 		if(shader->use_mis && shader->has_surface_emission) | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 			scene->light_manager->need_update = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* setup shader engine */ | 
					
						
							|  |  |  | 	og->ss = ss; | 
					
						
							| 
									
										
										
										
											2012-11-20 17:40:21 +00:00
										 |  |  | 	og->ts = ts; | 
					
						
							| 
									
										
										
										
											2012-09-03 18:51:02 +00:00
										 |  |  | 	og->services = services; | 
					
						
							| 
									
										
										
										
											2013-02-14 16:11:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	int background_id = scene->shader_manager->get_shader_id(scene->default_background); | 
					
						
							| 
									
										
										
										
											2012-09-03 18:50:59 +00:00
										 |  |  | 	og->background_state = og->surface_state[background_id & SHADER_MASK]; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	og->use = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	foreach(Shader *shader, scene->shaders) | 
					
						
							|  |  |  | 		shader->need_update = false; | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	need_update = false; | 
					
						
							| 
									
										
										
										
											2011-05-31 16:21:30 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	/* set texture system */ | 
					
						
							|  |  |  | 	scene->image_manager->set_osl_texture_system((void*)ts); | 
					
						
							| 
									
										
										
										
											2011-09-27 20:37:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	device_update_common(device, dscene, scene, progress); | 
					
						
							| 
									
										
										
										
											2013-06-28 13:05:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-01-03 18:28:33 +05:00
										 |  |  | 		/* Perform greedyjit optimization.
 | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * This might waste time on optimizing gorups which are never actually | 
					
						
							|  |  |  | 		 * used, but this prevents OSL from allocating data on TLS at render | 
					
						
							|  |  |  | 		 * time. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * This is much better for us because this way we aren't required to | 
					
						
							|  |  |  | 		 * stop task scheduler threads to make sure all TLS is clean and don't | 
					
						
							|  |  |  | 		 * have issues with TLS data free accessing freed memory if task scheduler | 
					
						
							|  |  |  | 		 * is being freed after the Session is freed. | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2013-06-28 13:05:21 +00:00
										 |  |  | 		thread_scoped_lock lock(ss_shared_mutex); | 
					
						
							|  |  |  | 		ss->optimize_all_groups(); | 
					
						
							| 
									
										
										
										
											2016-01-03 18:28:33 +05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 20:26:52 +00:00
										 |  |  | void OSLShaderManager::device_free(Device *device, DeviceScene *dscene, Scene *scene) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	OSLGlobals *og = (OSLGlobals*)device->osl_memory(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 20:26:52 +00:00
										 |  |  | 	device_free_common(device, dscene, scene); | 
					
						
							| 
									
										
										
										
											2011-09-27 20:37:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	/* clear shader engine */ | 
					
						
							|  |  |  | 	og->use = false; | 
					
						
							|  |  |  | 	og->ss = NULL; | 
					
						
							| 
									
										
										
										
											2012-11-20 17:40:21 +00:00
										 |  |  | 	og->ts = NULL; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	og->surface_state.clear(); | 
					
						
							|  |  |  | 	og->volume_state.clear(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	og->displacement_state.clear(); | 
					
						
							| 
									
										
										
										
											2017-07-08 02:46:06 +02:00
										 |  |  | 	og->bump_state.clear(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	og->background_state.reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | void OSLShaderManager::texture_system_init() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-04 23:48:07 +00:00
										 |  |  | 	/* create texture system, shared between different renders to reduce memory usage */ | 
					
						
							|  |  |  | 	thread_scoped_lock lock(ts_shared_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ts_shared_users == 0) { | 
					
						
							|  |  |  | 		ts_shared = TextureSystem::create(true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ts_shared->attribute("automip",  1); | 
					
						
							|  |  |  | 		ts_shared->attribute("autotile", 64); | 
					
						
							|  |  |  | 		ts_shared->attribute("gray_to_rgb", 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* effectively unlimited for now, until we support proper mipmap lookups */ | 
					
						
							|  |  |  | 		ts_shared->attribute("max_memory_MB", 16384); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ts = ts_shared; | 
					
						
							|  |  |  | 	ts_shared_users++; | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | void OSLShaderManager::texture_system_free() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* shared texture system decrease users and destroy if no longer used */ | 
					
						
							|  |  |  | 	thread_scoped_lock lock(ts_shared_mutex); | 
					
						
							|  |  |  | 	ts_shared_users--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ts_shared_users == 0) { | 
					
						
							| 
									
										
										
										
											2016-02-03 12:10:00 +01:00
										 |  |  | 		ts_shared->invalidate_all(true); | 
					
						
							|  |  |  | 		OSL::TextureSystem::destroy(ts_shared); | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 		ts_shared = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ts = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | void OSLShaderManager::shading_system_init() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 	/* create shading system, shared between different renders to reduce memory usage */ | 
					
						
							|  |  |  | 	thread_scoped_lock lock(ss_shared_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ss_shared_users == 0) { | 
					
						
							|  |  |  | 		services_shared = new OSLRenderServices(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-23 13:58:31 +01:00
										 |  |  | 		string shader_path = path_get("shader"); | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 		/* Annoying thing, Cycles stores paths in UTF-8 codepage, so it can
 | 
					
						
							|  |  |  | 		 * operate with file paths with any character. This requires to use wide | 
					
						
							|  |  |  | 		 * char functions, but OSL uses old fashioned ANSI functions which means: | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * - We have to convert our paths to ANSI before passing to OSL | 
					
						
							|  |  |  | 		 * - OSL can't be used when there's a multi-byte character in the path | 
					
						
							|  |  |  | 		 *   to the shaders folder. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		shader_path = string_to_ansi(shader_path); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-28 01:07:59 +01:00
										 |  |  | 		ss_shared = new OSL::ShadingSystem(services_shared, ts_shared, &errhandler); | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 		ss_shared->attribute("lockgeom", 1); | 
					
						
							|  |  |  | 		ss_shared->attribute("commonspace", "world"); | 
					
						
							| 
									
										
										
										
											2016-03-23 13:58:31 +01:00
										 |  |  | 		ss_shared->attribute("searchpath:shader", shader_path); | 
					
						
							| 
									
										
										
										
											2016-01-03 18:28:33 +05:00
										 |  |  | 		ss_shared->attribute("greedyjit", 1); | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-23 13:58:31 +01:00
										 |  |  | 		VLOG(1) << "Using shader search path: " << shader_path; | 
					
						
							| 
									
										
										
										
											2014-11-16 01:58:55 +05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 		/* our own ray types */ | 
					
						
							|  |  |  | 		static const char *raytypes[] = { | 
					
						
							| 
									
										
										
										
											2013-12-28 01:54:44 +01:00
										 |  |  | 			"camera",			/* PATH_RAY_CAMERA */ | 
					
						
							|  |  |  | 			"reflection",		/* PATH_RAY_REFLECT */ | 
					
						
							|  |  |  | 			"refraction",		/* PATH_RAY_TRANSMIT */ | 
					
						
							|  |  |  | 			"diffuse",			/* PATH_RAY_DIFFUSE */ | 
					
						
							|  |  |  | 			"glossy",			/* PATH_RAY_GLOSSY */ | 
					
						
							|  |  |  | 			"singular",			/* PATH_RAY_SINGULAR */ | 
					
						
							|  |  |  | 			"transparent",		/* PATH_RAY_TRANSPARENT */ | 
					
						
							| 
									
										
										
										
											2018-02-20 14:36:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-01 23:40:38 +02:00
										 |  |  | 			"shadow",			/* PATH_RAY_SHADOW_OPAQUE_NON_CATCHER */ | 
					
						
							|  |  |  | 			"shadow",			/* PATH_RAY_SHADOW_OPAQUE_CATCHER */ | 
					
						
							|  |  |  | 			"shadow",			/* PATH_RAY_SHADOW_TRANSPARENT_NON_CATCHER */ | 
					
						
							|  |  |  | 			"shadow",			/* PATH_RAY_SHADOW_TRANSPARENT_CATCHER */ | 
					
						
							| 
									
										
										
										
											2013-08-18 14:15:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							| 
									
										
										
										
											2018-02-20 14:36:50 +01:00
										 |  |  | 			"volume_scatter",	/* PATH_RAY_VOLUME_SCATTER */ | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-18 14:15:57 +00:00
										 |  |  | 			"__unused__", | 
					
						
							| 
									
										
										
										
											2015-05-08 12:10:26 +02:00
										 |  |  | 			"diffuse_ancestor",	/* PATH_RAY_DIFFUSE_ANCESTOR */ | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							| 
									
										
										
										
											2018-02-20 14:36:50 +01:00
										 |  |  | 			"__unused__", | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							| 
									
										
										
										
											2018-02-20 14:22:40 +01:00
										 |  |  | 			"__unused__", | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							|  |  |  | 			"__unused__", | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const int nraytypes = sizeof(raytypes)/sizeof(raytypes[0]); | 
					
						
							|  |  |  | 		ss_shared->attribute("raytypes", TypeDesc(TypeDesc::STRING, nraytypes), raytypes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		OSLShader::register_closures((OSLShadingSystem*)ss_shared); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		loaded_shaders.clear(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ss = ss_shared; | 
					
						
							|  |  |  | 	services = services_shared; | 
					
						
							|  |  |  | 	ss_shared_users++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLShaderManager::shading_system_free() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* shared shading system decrease users and destroy if no longer used */ | 
					
						
							|  |  |  | 	thread_scoped_lock lock(ss_shared_mutex); | 
					
						
							|  |  |  | 	ss_shared_users--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(ss_shared_users == 0) { | 
					
						
							| 
									
										
										
										
											2015-01-28 01:07:59 +01:00
										 |  |  | 		delete ss_shared; | 
					
						
							| 
									
										
										
										
											2013-04-22 14:27:12 +00:00
										 |  |  | 		ss_shared = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		delete services_shared; | 
					
						
							|  |  |  | 		services_shared = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ss = NULL; | 
					
						
							|  |  |  | 	services = NULL; | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool OSLShaderManager::osl_compile(const string& inputfile, const string& outputfile) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-01-17 00:13:30 +05:00
										 |  |  | 	vector<string> options; | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	string stdosl_path; | 
					
						
							| 
									
										
										
										
											2014-08-05 15:53:00 +06:00
										 |  |  | 	string shader_path = path_get("shader"); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* specify output file name */ | 
					
						
							|  |  |  | 	options.push_back("-o"); | 
					
						
							|  |  |  | 	options.push_back(outputfile); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* specify standard include path */ | 
					
						
							| 
									
										
										
										
											2015-02-05 12:06:24 +05:00
										 |  |  | 	string include_path_arg = string("-I") + shader_path; | 
					
						
							|  |  |  | 	options.push_back(include_path_arg); | 
					
						
							| 
									
										
										
										
											2014-08-05 11:11:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	stdosl_path = path_get("shader/stdosl.h"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* compile */ | 
					
						
							| 
									
										
										
										
											2016-01-16 01:32:22 +01:00
										 |  |  | 	OSL::OSLCompiler *compiler = new OSL::OSLCompiler(&OSL::ErrorHandler::default_handler()); | 
					
						
							| 
									
										
										
										
											2014-08-05 11:11:52 +02:00
										 |  |  | 	bool ok = compiler->compile(string_view(inputfile), options, string_view(stdosl_path)); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	delete compiler; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ok; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool OSLShaderManager::osl_query(OSL::OSLQuery& query, const string& filepath) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	string searchpath = path_user_get("shaders"); | 
					
						
							|  |  |  | 	return query.open(filepath, searchpath); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static string shader_filepath_hash(const string& filepath, uint64_t modified_time) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* compute a hash from filepath and modified time to detect changes */ | 
					
						
							|  |  |  | 	MD5Hash md5; | 
					
						
							|  |  |  | 	md5.append((const uint8_t*)filepath.c_str(), filepath.size()); | 
					
						
							|  |  |  | 	md5.append((const uint8_t*)&modified_time, sizeof(modified_time)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return md5.get_hex(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *OSLShaderManager::shader_test_loaded(const string& hash) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 	map<string, OSLShaderInfo>::iterator it = loaded_shaders.find(hash); | 
					
						
							|  |  |  | 	return (it == loaded_shaders.end())? NULL: it->first.c_str(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OSLShaderInfo *OSLShaderManager::shader_loaded_info(const string& hash) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	map<string, OSLShaderInfo>::iterator it = loaded_shaders.find(hash); | 
					
						
							|  |  |  | 	return (it == loaded_shaders.end())? NULL: &it->second; | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *OSLShaderManager::shader_load_filepath(string filepath) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	size_t len = filepath.size(); | 
					
						
							|  |  |  | 	string extension = filepath.substr(len - 4); | 
					
						
							|  |  |  | 	uint64_t modified_time = path_modified_time(filepath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(extension == ".osl") { | 
					
						
							|  |  |  | 		/* .OSL File */ | 
					
						
							|  |  |  | 		string osopath = filepath.substr(0, len - 4) + ".oso"; | 
					
						
							|  |  |  | 		uint64_t oso_modified_time = path_modified_time(osopath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* test if we have loaded the corresponding .OSO already */ | 
					
						
							|  |  |  | 		if(oso_modified_time != 0) { | 
					
						
							|  |  |  | 			const char *hash = shader_test_loaded(shader_filepath_hash(osopath, oso_modified_time)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(hash) | 
					
						
							|  |  |  | 				return hash; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* autocompile .OSL to .OSO if needed */ | 
					
						
							|  |  |  | 		if(oso_modified_time == 0 || (oso_modified_time < modified_time)) { | 
					
						
							|  |  |  | 			OSLShaderManager::osl_compile(filepath, osopath); | 
					
						
							|  |  |  | 			modified_time = path_modified_time(osopath); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			modified_time = oso_modified_time; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		filepath = osopath; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		if(extension == ".oso") { | 
					
						
							|  |  |  | 			/* .OSO File, nothing to do */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if(path_dirname(filepath) == "") { | 
					
						
							|  |  |  | 			/* .OSO File in search path */ | 
					
						
							|  |  |  | 			filepath = path_join(path_user_get("shaders"), filepath + ".oso"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			/* unknown file */ | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* test if we have loaded this .OSO already */ | 
					
						
							|  |  |  | 		const char *hash = shader_test_loaded(shader_filepath_hash(filepath, modified_time)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(hash) | 
					
						
							|  |  |  | 			return hash; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* read oso bytecode from file */ | 
					
						
							|  |  |  | 	string bytecode_hash = shader_filepath_hash(filepath, modified_time); | 
					
						
							|  |  |  | 	string bytecode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(!path_read_text(filepath, bytecode)) { | 
					
						
							|  |  |  | 		fprintf(stderr, "Cycles shader graph: failed to read file %s\n", filepath.c_str()); | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 		OSLShaderInfo info; | 
					
						
							|  |  |  | 		loaded_shaders[bytecode_hash] = info; /* to avoid repeat tries */ | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return shader_load_bytecode(bytecode_hash, bytecode); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *OSLShaderManager::shader_load_bytecode(const string& hash, const string& bytecode) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-09 14:13:43 +00:00
										 |  |  | 	ss->LoadMemoryCompiledShader(hash.c_str(), bytecode.c_str()); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 	OSLShaderInfo info; | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(!info.query.open_bytecode(bytecode)) { | 
					
						
							|  |  |  | 		fprintf(stderr, "OSL query error: %s\n", info.query.geterror().c_str()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* this is a bit weak, but works */ | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 	info.has_surface_emission = (bytecode.find("\"emission\"") != string::npos); | 
					
						
							|  |  |  | 	info.has_surface_transparent = (bytecode.find("\"transparent\"") != string::npos); | 
					
						
							| 
									
										
										
										
											2013-04-01 20:26:52 +00:00
										 |  |  | 	info.has_surface_bssrdf = (bytecode.find("\"bssrdf\"") != string::npos); | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 	loaded_shaders[hash] = info; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return loaded_shaders.find(hash)->first.c_str(); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | OSLNode *OSLShaderManager::osl_node(const std::string& filepath, | 
					
						
							|  |  |  |                                     const std::string& bytecode_hash, | 
					
						
							|  |  |  |                                     const std::string& bytecode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* create query */ | 
					
						
							|  |  |  | 	const char *hash; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(!filepath.empty()) { | 
					
						
							|  |  |  | 		hash = shader_load_filepath(filepath); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		hash = shader_test_loaded(bytecode_hash); | 
					
						
							|  |  |  | 		if(!hash) | 
					
						
							|  |  |  | 			hash = shader_load_bytecode(bytecode_hash, bytecode); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(!hash) { | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	OSLShaderInfo *info = shader_loaded_info(hash); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* count number of inputs */ | 
					
						
							|  |  |  | 	size_t num_inputs = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(int i = 0; i < info->query.nparams(); i++) { | 
					
						
							|  |  |  | 		const OSL::OSLQuery::Parameter *param = info->query.getparam(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* skip unsupported types */ | 
					
						
							|  |  |  | 		if(param->varlenarray || param->isstruct || param->type.arraylen > 1) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(!param->isoutput) | 
					
						
							|  |  |  | 			num_inputs++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create node */ | 
					
						
							|  |  |  | 	OSLNode *node = OSLNode::create(num_inputs); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* add new sockets from parameters */ | 
					
						
							|  |  |  | 	set<void*> used_sockets; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(int i = 0; i < info->query.nparams(); i++) { | 
					
						
							|  |  |  | 		const OSL::OSLQuery::Parameter *param = info->query.getparam(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* skip unsupported types */ | 
					
						
							|  |  |  | 		if(param->varlenarray || param->isstruct || param->type.arraylen > 1) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		SocketType::Type socket_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(param->isclosure) { | 
					
						
							|  |  |  | 			socket_type = SocketType::CLOSURE; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if(param->type.vecsemantics != TypeDesc::NOSEMANTICS) { | 
					
						
							|  |  |  | 			if(param->type.vecsemantics == TypeDesc::COLOR) | 
					
						
							|  |  |  | 				socket_type = SocketType::COLOR; | 
					
						
							|  |  |  | 			else if(param->type.vecsemantics == TypeDesc::POINT) | 
					
						
							|  |  |  | 				socket_type = SocketType::POINT; | 
					
						
							|  |  |  | 			else if(param->type.vecsemantics == TypeDesc::VECTOR) | 
					
						
							|  |  |  | 				socket_type = SocketType::VECTOR; | 
					
						
							|  |  |  | 			else if(param->type.vecsemantics == TypeDesc::NORMAL) | 
					
						
							|  |  |  | 				socket_type = SocketType::NORMAL; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(!param->isoutput && param->validdefault) { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 				float3 *default_value = (float3*)node->input_default_value(); | 
					
						
							|  |  |  | 				default_value->x = param->fdefault[0]; | 
					
						
							|  |  |  | 				default_value->y = param->fdefault[1]; | 
					
						
							|  |  |  | 				default_value->z = param->fdefault[2]; | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if(param->type.aggregate == TypeDesc::SCALAR) { | 
					
						
							|  |  |  | 			if(param->type.basetype == TypeDesc::INT) { | 
					
						
							|  |  |  | 				socket_type = SocketType::INT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(!param->isoutput && param->validdefault) { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					*(int*)node->input_default_value() = param->idefault[0]; | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if(param->type.basetype == TypeDesc::FLOAT) { | 
					
						
							|  |  |  | 				socket_type = SocketType::FLOAT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(!param->isoutput && param->validdefault) { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					*(float*)node->input_default_value() = param->fdefault[0]; | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if(param->type.basetype == TypeDesc::STRING) { | 
					
						
							|  |  |  | 				socket_type = SocketType::STRING; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(!param->isoutput && param->validdefault) { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					*(ustring*)node->input_default_value() = param->sdefault[0]; | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(param->isoutput) { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 			node->add_output(param->name, socket_type); | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 			node->add_input(param->name, socket_type); | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* set bytcode hash or filepath */ | 
					
						
							|  |  |  | 	if(!bytecode_hash.empty()) { | 
					
						
							|  |  |  | 		node->bytecode_hash = bytecode_hash; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		node->filepath = filepath; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 	/* Generate inputs and outputs */ | 
					
						
							|  |  |  | 	node->create_inputs_outputs(node->type); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-29 15:10:34 +02:00
										 |  |  | 	return node; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | /* Graph Compiler */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Cycles: Add Support for IES files as textures for light strength
This patch adds support for IES files, a file format that is commonly used to store the directional intensity distribution of light sources.
The new IES node is supposed to be plugged into the Strength input of the Emission node of the lamp.
Since people generating IES files do not really seem to care about the standard, the parser is flexible enough to accept all test files I have tried.
Some common weirdnesses are distributing values over multiple lines that should go into one line, using commas instead of spaces as delimiters and adding various useless stuff at the end of the file.
The user interface of the node is similar to the script node, the user can either select an internal Text or load a file.
Internally, IES files are handled similar to Image textures: They are stored in slots by the LightManager and each unique IES is assigned to one slot.
The local coordinate system of the lamp is used, so that the direction of the light can be changed. For UI reasons, it's usually best to add an area light,
rotate it and then change its type, since especially the point light does not immediately show its local coordinate system in the viewport.
Reviewers: #cycles, dingto, sergey, brecht
Reviewed By: #cycles, dingto, brecht
Subscribers: OgDEV, crazyrobinhood, secundar, cardboard, pisuke, intrah, swerner, micah_denn, harvester, gottfried, disnel, campbellbarton, duarteframos, Lapineige, brecht, juicyfruit, dingto, marek, rickyblender, bliblubli, lockal, sergey
Differential Revision: https://developer.blender.org/D1543
											
										 
											2018-05-27 00:46:37 +02:00
										 |  |  | OSLCompiler::OSLCompiler(void *manager_, void *shadingsys_, | 
					
						
							|  |  |  |                          ImageManager *image_manager_, | 
					
						
							|  |  |  |                          LightManager *light_manager_) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	manager = manager_; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	shadingsys = shadingsys_; | 
					
						
							| 
									
										
										
										
											2012-11-20 17:40:10 +00:00
										 |  |  | 	image_manager = image_manager_; | 
					
						
							| 
									
										
											  
											
												Cycles: Add Support for IES files as textures for light strength
This patch adds support for IES files, a file format that is commonly used to store the directional intensity distribution of light sources.
The new IES node is supposed to be plugged into the Strength input of the Emission node of the lamp.
Since people generating IES files do not really seem to care about the standard, the parser is flexible enough to accept all test files I have tried.
Some common weirdnesses are distributing values over multiple lines that should go into one line, using commas instead of spaces as delimiters and adding various useless stuff at the end of the file.
The user interface of the node is similar to the script node, the user can either select an internal Text or load a file.
Internally, IES files are handled similar to Image textures: They are stored in slots by the LightManager and each unique IES is assigned to one slot.
The local coordinate system of the lamp is used, so that the direction of the light can be changed. For UI reasons, it's usually best to add an area light,
rotate it and then change its type, since especially the point light does not immediately show its local coordinate system in the viewport.
Reviewers: #cycles, dingto, sergey, brecht
Reviewed By: #cycles, dingto, brecht
Subscribers: OgDEV, crazyrobinhood, secundar, cardboard, pisuke, intrah, swerner, micah_denn, harvester, gottfried, disnel, campbellbarton, duarteframos, Lapineige, brecht, juicyfruit, dingto, marek, rickyblender, bliblubli, lockal, sergey
Differential Revision: https://developer.blender.org/D1543
											
										 
											2018-05-27 00:46:37 +02:00
										 |  |  | 	light_manager = light_manager_; | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	current_type = SHADER_TYPE_SURFACE; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	current_shader = NULL; | 
					
						
							|  |  |  | 	background = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | string OSLCompiler::id(ShaderNode *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* assign layer unique name based on pointer address + bump mode */ | 
					
						
							|  |  |  | 	stringstream stream; | 
					
						
							| 
									
										
										
										
											2016-08-04 14:48:34 +03:00
										 |  |  | 	stream << "node_" << node->type->name << "_" << node; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return stream.str(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | string OSLCompiler::compatible_name(ShaderNode *node, ShaderInput *input) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 	string sname(input->name().string()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	size_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 	/* strip whitespace */ | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	while((i = sname.find(" ")) != string::npos) | 
					
						
							|  |  |  | 		sname.replace(i, 1, ""); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 	/* if output exists with the same name, add "In" suffix */ | 
					
						
							|  |  |  | 	foreach(ShaderOutput *output, node->outputs) { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 		if(input->name() == output->name()) { | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 			sname += "In"; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return sname; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | string OSLCompiler::compatible_name(ShaderNode *node, ShaderOutput *output) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 	string sname(output->name().string()); | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 	size_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* strip whitespace */ | 
					
						
							|  |  |  | 	while((i = sname.find(" ")) != string::npos) | 
					
						
							|  |  |  | 		sname.replace(i, 1, ""); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-11-20 17:40:21 +00:00
										 |  |  | 	/* if input exists with the same name, add "Out" suffix */ | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 	foreach(ShaderInput *input, node->inputs) { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 		if(input->name() == output->name()) { | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 			sname += "Out"; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	return sname; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool OSLCompiler::node_skip_input(ShaderNode *node, ShaderInput *input) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* exception for output node, only one input is actually used
 | 
					
						
							| 
									
										
										
										
											2012-06-09 17:22:52 +00:00
										 |  |  | 	 * depending on the current shader type */ | 
					
						
							| 
									
										
										
										
											2012-11-27 16:02:12 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 	if(input->flags() & SocketType::SVM_INTERNAL) | 
					
						
							| 
									
										
										
										
											2012-11-27 16:02:12 +00:00
										 |  |  | 		return true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-08 00:41:01 +02:00
										 |  |  | 	if(node->special_type == SHADER_SPECIAL_TYPE_OUTPUT) { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 		if(input->name() == "Surface" && current_type != SHADER_TYPE_SURFACE) | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 		if(input->name() == "Volume" && current_type != SHADER_TYPE_VOLUME) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 		if(input->name() == "Displacement" && current_type != SHADER_TYPE_DISPLACEMENT) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2016-08-14 11:44:25 -04:00
										 |  |  | 		if(input->name() == "Normal" && current_type != SHADER_TYPE_BUMP) | 
					
						
							| 
									
										
										
										
											2012-10-20 12:18:00 +00:00
										 |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-02 00:05:16 +02:00
										 |  |  | 	else if(node->special_type == SHADER_SPECIAL_TYPE_BUMP) { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 		if(input->name() == "Height") | 
					
						
							| 
									
										
										
										
											2012-10-20 15:09:36 +00:00
										 |  |  | 			return true; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-05-02 00:05:16 +02:00
										 |  |  | 	else if(current_type == SHADER_TYPE_DISPLACEMENT && input->link && input->link->parent->special_type == SHADER_SPECIAL_TYPE_BUMP) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | void OSLCompiler::add(ShaderNode *node, const char *name, bool isfilepath) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	/* load filepath */ | 
					
						
							|  |  |  | 	if(isfilepath) { | 
					
						
							|  |  |  | 		name = ((OSLShaderManager*)manager)->shader_load_filepath(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(name == NULL) | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	/* pass in fixed parameter values */ | 
					
						
							|  |  |  | 	foreach(ShaderInput *input, node->inputs) { | 
					
						
							|  |  |  | 		if(!input->link) { | 
					
						
							|  |  |  | 			/* checks to untangle graphs */ | 
					
						
							|  |  |  | 			if(node_skip_input(node, input)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			/* already has default value assigned */ | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 			else if(input->flags() & SocketType::DEFAULT_LINK_MASK) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 			string param_name = compatible_name(node, input); | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 			const SocketType& socket = input->socket_type; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 			switch(input->type()) { | 
					
						
							|  |  |  | 				case SocketType::COLOR: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter_color(param_name.c_str(), node->get_float3(socket)); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::POINT: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter_point(param_name.c_str(), node->get_float3(socket)); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::VECTOR: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter_vector(param_name.c_str(), node->get_float3(socket)); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::NORMAL: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter_normal(param_name.c_str(), node->get_float3(socket)); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::FLOAT: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter(param_name.c_str(), node->get_float(socket)); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::INT: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter(param_name.c_str(), node->get_int(socket)); | 
					
						
							| 
									
										
										
										
											2012-10-20 13:11:45 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::STRING: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 					parameter(param_name.c_str(), node->get_string(socket)); | 
					
						
							| 
									
										
										
										
											2012-11-06 21:36:44 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:32:09 +02:00
										 |  |  | 				case SocketType::CLOSURE: | 
					
						
							|  |  |  | 				case SocketType::UNDEFINED: | 
					
						
							|  |  |  | 				default: | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-06 21:02:41 +01:00
										 |  |  | 	/* create shader of the appropriate type. OSL only distinguishes between "surface"
 | 
					
						
							|  |  |  | 	 * and "displacement" atm */ | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	if(current_type == SHADER_TYPE_SURFACE) | 
					
						
							|  |  |  | 		ss->Shader("surface", name, id(node).c_str()); | 
					
						
							|  |  |  | 	else if(current_type == SHADER_TYPE_VOLUME) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		ss->Shader("surface", name, id(node).c_str()); | 
					
						
							|  |  |  | 	else if(current_type == SHADER_TYPE_DISPLACEMENT) | 
					
						
							| 
									
										
										
										
											2014-03-06 21:02:41 +01:00
										 |  |  | 		ss->Shader("displacement", name, id(node).c_str()); | 
					
						
							| 
									
										
										
										
											2016-08-14 11:44:25 -04:00
										 |  |  | 	else if(current_type == SHADER_TYPE_BUMP) | 
					
						
							|  |  |  | 		ss->Shader("displacement", name, id(node).c_str()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		assert(0); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* link inputs to other nodes */ | 
					
						
							|  |  |  | 	foreach(ShaderInput *input, node->inputs) { | 
					
						
							|  |  |  | 		if(input->link) { | 
					
						
							|  |  |  | 			if(node_skip_input(node, input)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* connect shaders */ | 
					
						
							|  |  |  | 			string id_from = id(input->link->parent); | 
					
						
							|  |  |  | 			string id_to = id(node); | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 			string param_from = compatible_name(input->link->parent, input->link); | 
					
						
							|  |  |  | 			string param_to = compatible_name(node, input); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			ss->ConnectShaders(id_from.c_str(), param_from.c_str(), id_to.c_str(), param_to.c_str()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* test if we shader contains specific closures */ | 
					
						
							|  |  |  | 	OSLShaderInfo *info = ((OSLShaderManager*)manager)->shader_loaded_info(name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-05 22:13:51 +01:00
										 |  |  | 	if(current_type == SHADER_TYPE_SURFACE) { | 
					
						
							|  |  |  | 		if(info) { | 
					
						
							|  |  |  | 			if(info->has_surface_emission) | 
					
						
							|  |  |  | 				current_shader->has_surface_emission = true; | 
					
						
							|  |  |  | 			if(info->has_surface_transparent) | 
					
						
							|  |  |  | 				current_shader->has_surface_transparent = true; | 
					
						
							|  |  |  | 			if(info->has_surface_bssrdf) { | 
					
						
							|  |  |  | 				current_shader->has_surface_bssrdf = true; | 
					
						
							|  |  |  | 				current_shader->has_bssrdf_bump = true; /* can't detect yet */ | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-08-20 03:25:13 +02:00
										 |  |  | 			current_shader->has_bump = true; /* can't detect yet */ | 
					
						
							| 
									
										
										
										
											2016-02-05 22:13:51 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(node->has_spatial_varying()) { | 
					
						
							|  |  |  | 			current_shader->has_surface_spatial_varying = true; | 
					
						
							| 
									
										
										
										
											2013-08-18 14:15:57 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-12-12 06:51:06 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-04-04 13:20:46 +02:00
										 |  |  | 	else if(current_type == SHADER_TYPE_VOLUME) { | 
					
						
							|  |  |  | 		if(node->has_spatial_varying()) | 
					
						
							| 
									
										
										
										
											2016-02-05 21:33:37 +01:00
										 |  |  | 			current_shader->has_volume_spatial_varying = true; | 
					
						
							| 
									
										
										
										
											2014-04-04 13:20:46 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-01-21 22:19:31 +05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(node->has_object_dependency()) { | 
					
						
							|  |  |  | 		current_shader->has_object_dependency = true; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-11-20 18:18:27 +05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-18 03:20:39 +01:00
										 |  |  | 	if(node->has_attribute_dependency()) { | 
					
						
							|  |  |  | 		current_shader->has_attribute_dependency = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-20 18:18:27 +05:00
										 |  |  | 	if(node->has_integrator_dependency()) { | 
					
						
							|  |  |  | 		current_shader->has_integrator_dependency = true; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | static TypeDesc array_typedesc(TypeDesc typedesc, int arraylength) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return TypeDesc((TypeDesc::BASETYPE)typedesc.basetype, | 
					
						
							|  |  |  | 	                (TypeDesc::AGGREGATE)typedesc.aggregate, | 
					
						
							|  |  |  | 	                (TypeDesc::VECSEMANTICS)typedesc.vecsemantics, | 
					
						
							|  |  |  | 	                arraylength); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(ShaderNode* node, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ustring uname = ustring(name); | 
					
						
							|  |  |  | 	const SocketType& socket = *(node->type->find_input(uname)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch(socket.type) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		case SocketType::BOOLEAN: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			int value = node->get_bool(socket); | 
					
						
							|  |  |  | 			ss->Parameter(name, TypeDesc::TypeInt, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::FLOAT: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			float value = node->get_float(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeFloat, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::INT: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			int value = node->get_int(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeInt, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::COLOR: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			float3 value = node->get_float3(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeColor, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::VECTOR: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			float3 value = node->get_float3(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeVector, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::POINT: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			float3 value = node->get_float3(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypePoint, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::NORMAL: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			float3 value = node->get_float3(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeNormal, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::POINT2: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			float2 value = node->get_float2(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc(TypeDesc::FLOAT, TypeDesc::VEC2, TypeDesc::POINT), &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::STRING: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			ustring value = node->get_string(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeString, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::ENUM: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			ustring value = node->get_string(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeString, &value); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::TRANSFORM: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Transform value = node->get_transform(socket); | 
					
						
							| 
									
										
										
										
											2018-03-08 06:48:14 +01:00
										 |  |  | 			ProjectionTransform projection(value); | 
					
						
							|  |  |  | 			projection = projection_transpose(projection); | 
					
						
							|  |  |  | 			ss->Parameter(uname, TypeDesc::TypeMatrix, &projection); | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::BOOLEAN_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			// OSL does not support booleans, so convert to int
 | 
					
						
							|  |  |  | 			const array<bool>& value = node->get_bool_array(socket); | 
					
						
							|  |  |  | 			array<int> intvalue(value.size()); | 
					
						
							| 
									
										
										
										
											2016-10-24 12:26:12 +02:00
										 |  |  | 			for(size_t i = 0; i < value.size(); i++) | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 				intvalue[i] = value[i]; | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(TypeDesc::TypeInt, value.size()), intvalue.data()); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::FLOAT_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			const array<float>& value = node->get_float_array(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(TypeDesc::TypeFloat, value.size()), value.data()); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::INT_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			const array<int>& value = node->get_int_array(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(TypeDesc::TypeInt, value.size()), value.data()); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::COLOR_ARRAY: | 
					
						
							|  |  |  | 		case SocketType::VECTOR_ARRAY: | 
					
						
							|  |  |  | 		case SocketType::POINT_ARRAY: | 
					
						
							|  |  |  | 		case SocketType::NORMAL_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			TypeDesc typedesc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			switch(socket.type) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				case SocketType::COLOR_ARRAY: typedesc = TypeDesc::TypeColor; break; | 
					
						
							|  |  |  | 				case SocketType::VECTOR_ARRAY: typedesc = TypeDesc::TypeVector; break; | 
					
						
							|  |  |  | 				case SocketType::POINT_ARRAY: typedesc = TypeDesc::TypePoint; break; | 
					
						
							|  |  |  | 				case SocketType::NORMAL_ARRAY: typedesc = TypeDesc::TypeNormal; break; | 
					
						
							|  |  |  | 				default: assert(0); break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// convert to tightly packed array since float3 has padding
 | 
					
						
							|  |  |  | 			const array<float3>& value = node->get_float3_array(socket); | 
					
						
							|  |  |  | 			array<float> fvalue(value.size() * 3); | 
					
						
							| 
									
										
										
										
											2016-10-24 12:26:12 +02:00
										 |  |  | 			for(size_t i = 0, j = 0; i < value.size(); i++) { | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 				fvalue[j++] = value[i].x; | 
					
						
							|  |  |  | 				fvalue[j++] = value[i].y; | 
					
						
							|  |  |  | 				fvalue[j++] = value[i].z; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(typedesc, value.size()), fvalue.data()); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::POINT2_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			const array<float2>& value = node->get_float2_array(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(TypeDesc(TypeDesc::FLOAT, TypeDesc::VEC2, TypeDesc::POINT), value.size()), value.data()); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::STRING_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			const array<ustring>& value = node->get_string_array(socket); | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(TypeDesc::TypeString, value.size()), value.data()); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::TRANSFORM_ARRAY: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			const array<Transform>& value = node->get_transform_array(socket); | 
					
						
							| 
									
										
										
										
											2018-03-08 06:48:14 +01:00
										 |  |  | 			array<ProjectionTransform> fvalue(value.size()); | 
					
						
							|  |  |  | 			for(size_t i = 0; i < value.size(); i++) { | 
					
						
							|  |  |  | 				fvalue[i] = projection_transpose(ProjectionTransform(value[i])); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			ss->Parameter(uname, array_typedesc(TypeDesc::TypeMatrix, fvalue.size()), fvalue.data()); | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case SocketType::CLOSURE: | 
					
						
							|  |  |  | 		case SocketType::NODE: | 
					
						
							|  |  |  | 		case SocketType::NODE_ARRAY: | 
					
						
							|  |  |  | 		case SocketType::UNDEFINED: | 
					
						
							| 
									
										
										
										
											2016-06-13 10:14:18 +02:00
										 |  |  | 		case SocketType::UINT: | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			assert(0); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | void OSLCompiler::parameter(const char *name, float f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeFloat, &f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_color(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeColor, &f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_point(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypePoint, &f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_normal(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeNormal, &f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_vector(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeVector, &f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, int f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeInt, &f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, const char *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeString, &s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, ustring s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	const char *str = s.c_str(); | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeString, &str); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, const Transform& tfm) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							| 
									
										
										
										
											2018-03-08 06:48:14 +01:00
										 |  |  | 	ProjectionTransform projection(tfm); | 
					
						
							|  |  |  | 	projection = projection_transpose(projection); | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeMatrix, (float*)&projection); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 15:41:37 +00:00
										 |  |  | void OSLCompiler::parameter_array(const char *name, const float f[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeFloat; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-08 01:54:35 +02:00
										 |  |  | void OSLCompiler::parameter_color_array(const char *name, const array<float3>& f) | 
					
						
							| 
									
										
										
										
											2012-09-15 15:41:37 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-05-08 01:54:35 +02:00
										 |  |  | 	/* NB: cycles float3 type is actually 4 floats! need to use an explicit array */ | 
					
						
							|  |  |  | 	array<float[3]> table(f.size()); | 
					
						
							| 
									
										
										
										
											2012-09-15 15:41:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-08 01:54:35 +02:00
										 |  |  | 	for(int i = 0; i < f.size(); ++i) { | 
					
						
							|  |  |  | 		table[i][0] = f[i].x; | 
					
						
							|  |  |  | 		table[i][1] = f[i].y; | 
					
						
							|  |  |  | 		table[i][2] = f[i].z; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-09-15 15:41:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							| 
									
										
										
										
											2016-05-08 01:54:35 +02:00
										 |  |  | 	TypeDesc type = TypeDesc::TypeColor; | 
					
						
							|  |  |  | 	type.arraylen = table.size(); | 
					
						
							|  |  |  | 	ss->Parameter(name, type, table.data()); | 
					
						
							| 
									
										
										
										
											2012-09-15 15:41:37 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-18 03:20:39 +01:00
										 |  |  | void OSLCompiler::parameter_attribute(const char *name, ustring s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(Attribute::name_standard(s.c_str())) | 
					
						
							|  |  |  | 		parameter(name, (string("geom:") + s.c_str()).c_str()); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		parameter(name, s.c_str()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-25 13:07:29 +05:00
										 |  |  | void OSLCompiler::find_dependencies(ShaderNodeSet& dependencies, ShaderInput *input) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	ShaderNode *node = (input->link)? input->link->parent: NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-25 13:46:51 +05:00
										 |  |  | 	if(node != NULL && dependencies.find(node) == dependencies.end()) { | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		foreach(ShaderInput *in, node->inputs) | 
					
						
							|  |  |  | 			if(!node_skip_input(node, in)) | 
					
						
							|  |  |  | 				find_dependencies(dependencies, in); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		dependencies.insert(node); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-25 13:07:29 +05:00
										 |  |  | void OSLCompiler::generate_nodes(const ShaderNodeSet& nodes) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-11-25 13:07:29 +05:00
										 |  |  | 	ShaderNodeSet done; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	bool nodes_done; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		nodes_done = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		foreach(ShaderNode *node, nodes) { | 
					
						
							|  |  |  | 			if(done.find(node) == done.end()) { | 
					
						
							|  |  |  | 				bool inputs_done = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				foreach(ShaderInput *input, node->inputs) | 
					
						
							|  |  |  | 					if(!node_skip_input(node, input)) | 
					
						
							|  |  |  | 						if(input->link && done.find(input->link->parent) == done.end()) | 
					
						
							|  |  |  | 							inputs_done = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(inputs_done) { | 
					
						
							|  |  |  | 					node->compile(*this); | 
					
						
							|  |  |  | 					done.insert(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-29 13:03:48 +01:00
										 |  |  | 					if(current_type == SHADER_TYPE_SURFACE) { | 
					
						
							|  |  |  | 						if(node->has_surface_emission()) | 
					
						
							|  |  |  | 							current_shader->has_surface_emission = true; | 
					
						
							|  |  |  | 						if(node->has_surface_transparent()) | 
					
						
							|  |  |  | 							current_shader->has_surface_transparent = true; | 
					
						
							| 
									
										
										
										
											2016-02-05 22:13:51 +01:00
										 |  |  | 						if(node->has_spatial_varying()) | 
					
						
							|  |  |  | 							current_shader->has_surface_spatial_varying = true; | 
					
						
							| 
									
										
										
										
											2014-03-29 13:03:48 +01:00
										 |  |  | 						if(node->has_surface_bssrdf()) { | 
					
						
							|  |  |  | 							current_shader->has_surface_bssrdf = true; | 
					
						
							|  |  |  | 							if(node->has_bssrdf_bump()) | 
					
						
							|  |  |  | 								current_shader->has_bssrdf_bump = true; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2017-08-20 03:25:13 +02:00
										 |  |  | 						if(node->has_bump()) { | 
					
						
							|  |  |  | 							current_shader->has_bump = true; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2013-08-18 14:15:57 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2014-04-04 13:20:46 +02:00
										 |  |  | 					else if(current_type == SHADER_TYPE_VOLUME) { | 
					
						
							|  |  |  | 						if(node->has_spatial_varying()) | 
					
						
							| 
									
										
										
										
											2016-02-05 21:33:37 +01:00
										 |  |  | 							current_shader->has_volume_spatial_varying = true; | 
					
						
							| 
									
										
										
										
											2014-04-04 13:20:46 +02:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					nodes_done = false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} while(!nodes_done); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-07 13:15:30 +05:00
										 |  |  | OSL::ShaderGroupRef OSLCompiler::compile_type(Shader *shader, ShaderGraph *graph, ShaderType type) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	current_type = type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-07 13:15:30 +05:00
										 |  |  | 	OSL::ShaderGroupRef group = ss->ShaderGroupBegin(shader->name.c_str()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ShaderNode *output = graph->output(); | 
					
						
							| 
									
										
										
										
											2015-11-25 13:07:29 +05:00
										 |  |  | 	ShaderNodeSet dependencies; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	if(type == SHADER_TYPE_SURFACE) { | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		/* generate surface shader */ | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 		find_dependencies(dependencies, output->input("Surface")); | 
					
						
							|  |  |  | 		generate_nodes(dependencies); | 
					
						
							|  |  |  | 		output->compile(*this); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-14 11:44:25 -04:00
										 |  |  | 	else if(type == SHADER_TYPE_BUMP) { | 
					
						
							|  |  |  | 		/* generate bump shader */ | 
					
						
							|  |  |  | 		find_dependencies(dependencies, output->input("Normal")); | 
					
						
							|  |  |  | 		generate_nodes(dependencies); | 
					
						
							|  |  |  | 		output->compile(*this); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	else if(type == SHADER_TYPE_VOLUME) { | 
					
						
							|  |  |  | 		/* generate volume shader */ | 
					
						
							|  |  |  | 		find_dependencies(dependencies, output->input("Volume")); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		generate_nodes(dependencies); | 
					
						
							|  |  |  | 		output->compile(*this); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if(type == SHADER_TYPE_DISPLACEMENT) { | 
					
						
							|  |  |  | 		/* generate displacement shader */ | 
					
						
							|  |  |  | 		find_dependencies(dependencies, output->input("Displacement")); | 
					
						
							|  |  |  | 		generate_nodes(dependencies); | 
					
						
							|  |  |  | 		output->compile(*this); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		assert(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ss->ShaderGroupEnd(); | 
					
						
							| 
									
										
										
										
											2013-12-08 15:15:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return group; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-20 18:18:27 +05:00
										 |  |  | void OSLCompiler::compile(Scene *scene, OSLGlobals *og, Shader *shader) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	if(shader->need_update) { | 
					
						
							|  |  |  | 		ShaderGraph *graph = shader->graph; | 
					
						
							|  |  |  | 		ShaderNode *output = (graph)? graph->output(): NULL; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-20 14:02:16 +02:00
										 |  |  | 		bool has_bump = (shader->displacement_method != DISPLACE_TRUE) && | 
					
						
							|  |  |  | 		                output->input("Surface")->link && output->input("Displacement")->link; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		/* finalize */ | 
					
						
							| 
									
										
										
										
											2015-11-20 18:18:27 +05:00
										 |  |  | 		shader->graph->finalize(scene, | 
					
						
							| 
									
										
										
										
											2017-08-20 14:02:16 +02:00
										 |  |  | 		                        has_bump, | 
					
						
							|  |  |  | 		                        shader->has_integrator_dependency, | 
					
						
							|  |  |  | 		                        shader->displacement_method == DISPLACE_BOTH); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		current_shader = shader; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		shader->has_surface = false; | 
					
						
							|  |  |  | 		shader->has_surface_emission = false; | 
					
						
							|  |  |  | 		shader->has_surface_transparent = false; | 
					
						
							| 
									
										
										
										
											2013-04-01 20:26:52 +00:00
										 |  |  | 		shader->has_surface_bssrdf = false; | 
					
						
							| 
									
										
										
										
											2017-08-20 14:02:16 +02:00
										 |  |  | 		shader->has_bump = has_bump; | 
					
						
							|  |  |  | 		shader->has_bssrdf_bump = has_bump; | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		shader->has_volume = false; | 
					
						
							|  |  |  | 		shader->has_displacement = false; | 
					
						
							| 
									
										
										
										
											2016-02-05 22:13:51 +01:00
										 |  |  | 		shader->has_surface_spatial_varying = false; | 
					
						
							| 
									
										
										
										
											2016-02-05 21:33:37 +01:00
										 |  |  | 		shader->has_volume_spatial_varying = false; | 
					
						
							| 
									
										
										
										
											2015-01-21 22:19:31 +05:00
										 |  |  | 		shader->has_object_dependency = false; | 
					
						
							| 
									
										
										
										
											2018-02-18 03:20:39 +01:00
										 |  |  | 		shader->has_attribute_dependency = false; | 
					
						
							| 
									
										
										
										
											2015-11-20 18:18:27 +05:00
										 |  |  | 		shader->has_integrator_dependency = false; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		/* generate surface shader */ | 
					
						
							|  |  |  | 		if(shader->used && graph && output->input("Surface")->link) { | 
					
						
							| 
									
										
										
										
											2013-12-08 15:15:27 +01:00
										 |  |  | 			shader->osl_surface_ref = compile_type(shader, shader->graph, SHADER_TYPE_SURFACE); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-20 14:02:16 +02:00
										 |  |  | 			if(has_bump) | 
					
						
							|  |  |  | 				shader->osl_surface_bump_ref = compile_type(shader, shader->graph, SHADER_TYPE_BUMP); | 
					
						
							| 
									
										
										
										
											2013-06-28 13:05:21 +00:00
										 |  |  | 			else | 
					
						
							| 
									
										
										
										
											2016-08-14 11:44:25 -04:00
										 |  |  | 				shader->osl_surface_bump_ref = OSL::ShaderGroupRef(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 			shader->has_surface = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2016-01-07 13:15:30 +05:00
										 |  |  | 			shader->osl_surface_ref = OSL::ShaderGroupRef(); | 
					
						
							|  |  |  | 			shader->osl_surface_bump_ref = OSL::ShaderGroupRef(); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		/* generate volume shader */ | 
					
						
							|  |  |  | 		if(shader->used && graph && output->input("Volume")->link) { | 
					
						
							| 
									
										
										
										
											2013-12-08 15:15:27 +01:00
										 |  |  | 			shader->osl_volume_ref = compile_type(shader, shader->graph, SHADER_TYPE_VOLUME); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 			shader->has_volume = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2016-01-07 13:15:30 +05:00
										 |  |  | 			shader->osl_volume_ref = OSL::ShaderGroupRef(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 		/* generate displacement shader */ | 
					
						
							|  |  |  | 		if(shader->used && graph && output->input("Displacement")->link) { | 
					
						
							| 
									
										
										
										
											2013-12-08 15:15:27 +01:00
										 |  |  | 			shader->osl_displacement_ref = compile_type(shader, shader->graph, SHADER_TYPE_DISPLACEMENT); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 			shader->has_displacement = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2016-01-07 13:15:30 +05:00
										 |  |  | 			shader->osl_displacement_ref = OSL::ShaderGroupRef(); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	/* push state to array for lookup */ | 
					
						
							| 
									
										
										
										
											2016-08-14 11:44:25 -04:00
										 |  |  | 	og->surface_state.push_back(shader->osl_surface_ref); | 
					
						
							| 
									
										
										
										
											2012-11-03 14:32:35 +00:00
										 |  |  | 	og->volume_state.push_back(shader->osl_volume_ref); | 
					
						
							|  |  |  | 	og->displacement_state.push_back(shader->osl_displacement_ref); | 
					
						
							| 
									
										
										
										
											2016-08-14 11:44:25 -04:00
										 |  |  | 	og->bump_state.push_back(shader->osl_surface_bump_ref); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::add(ShaderNode * /*node*/, const char * /*name*/, bool /*isfilepath*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-07 19:48:28 +02:00
										 |  |  | void OSLCompiler::parameter(ShaderNode * /*node*/, const char * /*name*/) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter(const char * /*name*/, float /*f*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter_color(const char * /*name*/, float3 /*f*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter_vector(const char * /*name*/, float3 /*f*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter_point(const char * /*name*/, float3 /*f*/) | 
					
						
							| 
									
										
										
										
											2012-09-15 18:08:51 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter_normal(const char * /*name*/, float3 /*f*/) | 
					
						
							| 
									
										
										
										
											2012-09-15 18:08:51 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter(const char * /*name*/, int /*f*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter(const char * /*name*/, const char * /*s*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter(const char * /*name*/, ustring /*s*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter(const char * /*name*/, const Transform& /*tfm*/) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 19:09:41 +05:00
										 |  |  | void OSLCompiler::parameter_array(const char * /*name*/, const float /*f*/[], int /*arraylen*/) | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:11 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-18 09:12:47 +02:00
										 |  |  | void OSLCompiler::parameter_color_array(const char * /*name*/, const array<float3>& /*f*/) | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:11 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | #endif /* WITH_OSL */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CCL_NAMESPACE_END | 
					
						
							|  |  |  | 
 |