| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright 2011, Blender Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "device.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "graph.h"
 | 
					
						
							|  |  |  | #include "light.h"
 | 
					
						
							|  |  |  | #include "osl.h"
 | 
					
						
							|  |  |  | #include "scene.h"
 | 
					
						
							|  |  |  | #include "shader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef WITH_OSL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "osl_globals.h"
 | 
					
						
							|  |  |  | #include "osl_services.h"
 | 
					
						
							|  |  |  | #include "osl_shader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "util_foreach.h"
 | 
					
						
							|  |  |  | #include "util_path.h"
 | 
					
						
							|  |  |  | #include "util_progress.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CCL_NAMESPACE_BEGIN | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef WITH_OSL
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Shader Manager */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OSLShaderManager::OSLShaderManager() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	services = new OSLRenderServices(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* if we let OSL create it, it leaks */ | 
					
						
							|  |  |  | 	ts = TextureSystem::create(true); | 
					
						
							|  |  |  | 	ts->attribute("automip",  1); | 
					
						
							|  |  |  | 	ts->attribute("autotile", 64); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ss = OSL::ShadingSystem::create(services, ts, &errhandler); | 
					
						
							|  |  |  | 	ss->attribute("lockgeom", 1); | 
					
						
							|  |  |  | 	ss->attribute("commonspace", "world"); | 
					
						
							|  |  |  | 	ss->attribute("optimize", 2); | 
					
						
							|  |  |  | 	//ss->attribute("debug", 1);
 | 
					
						
							|  |  |  | 	//ss->attribute("statistics:level", 1);
 | 
					
						
							|  |  |  | 	ss->attribute("searchpath:shader", path_get("shader").c_str()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	OSLShader::register_closures(ss); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OSLShaderManager::~OSLShaderManager() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem::destroy(ss); | 
					
						
							|  |  |  | 	OSL::TextureSystem::destroy(ts); | 
					
						
							|  |  |  | 	delete services; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLShaderManager::device_update(Device *device, DeviceScene *dscene, Scene *scene, Progress& progress) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* test if we need to update */ | 
					
						
							|  |  |  | 	bool need_update = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	foreach(Shader *shader, scene->shaders) | 
					
						
							|  |  |  | 		if(shader->need_update) | 
					
						
							|  |  |  | 			need_update = true; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if(!need_update) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	device_free(device, dscene); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create shaders */ | 
					
						
							|  |  |  | 	OSLGlobals *og = (OSLGlobals*)device->osl_memory(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	foreach(Shader *shader, scene->shaders) { | 
					
						
							|  |  |  | 		assert(shader->graph); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(progress.get_cancel()) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 20:37:24 +00:00
										 |  |  | 		if(shader->sample_as_light && shader->has_surface_emission) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 			scene->light_manager->need_update = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		OSLCompiler compiler((void*)ss); | 
					
						
							|  |  |  | 		compiler.background = (shader == scene->shaders[scene->default_background]); | 
					
						
							|  |  |  | 		compiler.compile(og, shader); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* setup shader engine */ | 
					
						
							|  |  |  | 	og->ss = ss; | 
					
						
							| 
									
										
										
										
											2012-09-03 18:51:02 +00:00
										 |  |  | 	og->services = services; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tls_create(OSLGlobals::ThreadData, og->thread_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	foreach(Shader *shader, scene->shaders) | 
					
						
							|  |  |  | 		shader->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); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLShaderManager::device_free(Device *device, DeviceScene *dscene) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSLGlobals *og = (OSLGlobals*)device->osl_memory(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-27 20:37:24 +00:00
										 |  |  | 	device_free_common(device, dscene); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	/* clear shader engine */ | 
					
						
							|  |  |  | 	og->use = false; | 
					
						
							|  |  |  | 	og->ss = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tls_delete(OSLGlobals::ThreadData, og->thread_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  | 	og->background_state.reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Graph Compiler */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OSLCompiler::OSLCompiler(void *shadingsys_) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	shadingsys = shadingsys_; | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 	stream << "node_" << node->name << "_" << node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 	string sname(input->name); | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  | 		if (strcmp(input->name, output->name)==0) { | 
					
						
							|  |  |  | 			sname += "In"; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return sname; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | string OSLCompiler::compatible_name(ShaderNode *node, ShaderOutput *output) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	string sname(output->name); | 
					
						
							|  |  |  | 	size_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* strip whitespace */ | 
					
						
							|  |  |  | 	while((i = sname.find(" ")) != string::npos) | 
					
						
							|  |  |  | 		sname.replace(i, 1, ""); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	/* if output exists with the same name, add "In" suffix */ | 
					
						
							|  |  |  | 	foreach(ShaderInput *input, node->inputs) { | 
					
						
							|  |  |  | 		if (strcmp(input->name, output->name)==0) { | 
					
						
							|  |  |  | 			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 */ | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(node->name == ustring("output")) { | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 		if(strcmp(input->name, "Surface") == 0 && current_type != SHADER_TYPE_SURFACE) | 
					
						
							|  |  |  | 			return true; | 
					
						
							|  |  |  | 		if(strcmp(input->name, "Volume") == 0 && current_type != SHADER_TYPE_VOLUME) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 			return true; | 
					
						
							|  |  |  | 		if(strcmp(input->name, "Displacement") == 0 && current_type != SHADER_TYPE_DISPLACEMENT) | 
					
						
							|  |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2012-10-20 12:18:00 +00:00
										 |  |  | 		if(strcmp(input->name, "Normal") == 0) | 
					
						
							|  |  |  | 			return true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else if(current_type == SHADER_TYPE_DISPLACEMENT && input->link && input->link->parent->name == ustring("bump")) | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::add(ShaderNode *node, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* 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 */ | 
					
						
							|  |  |  | 			else if(input->default_value != ShaderInput::NONE) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 			string param_name = compatible_name(node, input); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 			switch(input->type) { | 
					
						
							|  |  |  | 				case SHADER_SOCKET_COLOR: | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 					parameter_color(param_name.c_str(), input->value); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case SHADER_SOCKET_POINT: | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 					parameter_point(param_name.c_str(), input->value); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case SHADER_SOCKET_VECTOR: | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 					parameter_vector(param_name.c_str(), input->value); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case SHADER_SOCKET_NORMAL: | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 					parameter_normal(param_name.c_str(), input->value); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case SHADER_SOCKET_FLOAT: | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:07 +00:00
										 |  |  | 					parameter(param_name.c_str(), input->value.x); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case SHADER_SOCKET_CLOSURE: | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* create shader of the appropriate type. we pass "surface" to all shaders,
 | 
					
						
							|  |  |  | 	 * because "volume" and "displacement" don't work yet in OSL. the shaders | 
					
						
							|  |  |  | 	 * work fine, but presumably these values would be used for more strict | 
					
						
							|  |  |  | 	 * checking, so when that is fixed, we should update the code here too. */ | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | 		ss->Shader("surface", name, id(node).c_str()); | 
					
						
							|  |  |  | 	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()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							|  |  |  | 	ss->Parameter(name, TypeDesc::TypeMatrix, (float*)&tfm); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_color_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeColor; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_vector_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeVector; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_normal_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeNormal; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_point_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypePoint; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_array(const char *name, const int f[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeInt; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_array(const char *name, const char * const s[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeString; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, s); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_array(const char *name, const Transform tfm[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	TypeDesc type = TypeDesc::TypeMatrix; | 
					
						
							|  |  |  | 	type.arraylen = arraylen; | 
					
						
							|  |  |  | 	ss->Parameter(name, type, (const float *)tfm); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | void OSLCompiler::find_dependencies(set<ShaderNode*>& dependencies, ShaderInput *input) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ShaderNode *node = (input->link)? input->link->parent: NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(node) { | 
					
						
							|  |  |  | 		foreach(ShaderInput *in, node->inputs) | 
					
						
							|  |  |  | 			if(!node_skip_input(node, in)) | 
					
						
							|  |  |  | 				find_dependencies(dependencies, in); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		dependencies.insert(node); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::generate_nodes(const set<ShaderNode*>& nodes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	set<ShaderNode*> done; | 
					
						
							|  |  |  | 	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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if(node->name == ustring("emission")) | 
					
						
							|  |  |  | 						current_shader->has_surface_emission = true; | 
					
						
							| 
									
										
										
										
											2011-09-27 20:37:24 +00:00
										 |  |  | 					if(node->name == ustring("transparent")) | 
					
						
							|  |  |  | 						current_shader->has_surface_transparent = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					nodes_done = false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} while(!nodes_done); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::compile_type(Shader *shader, ShaderGraph *graph, ShaderType type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	current_type = type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ss->ShaderGroupBegin(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ShaderNode *output = graph->output(); | 
					
						
							|  |  |  | 	set<ShaderNode*> dependencies; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	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(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::compile(OSLGlobals *og, Shader *shader) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	OSL::ShadingSystem *ss = (OSL::ShadingSystem*)shadingsys; | 
					
						
							|  |  |  | 	ShaderGraph *graph = shader->graph; | 
					
						
							|  |  |  | 	ShaderNode *output = (graph)? graph->output(): NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* copy graph for shader with bump mapping */ | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	if(output->input("Surface")->link && output->input("Displacement")->link) | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		if(!shader->graph_bump) | 
					
						
							|  |  |  | 			shader->graph_bump = shader->graph->copy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* finalize */ | 
					
						
							|  |  |  | 	shader->graph->finalize(false, true); | 
					
						
							|  |  |  | 	if(shader->graph_bump) | 
					
						
							|  |  |  | 		shader->graph_bump->finalize(true, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	current_shader = shader; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	shader->has_surface = false; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	shader->has_surface_emission = false; | 
					
						
							| 
									
										
										
										
											2011-09-27 20:37:24 +00:00
										 |  |  | 	shader->has_surface_transparent = false; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	shader->has_volume = false; | 
					
						
							|  |  |  | 	shader->has_displacement = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* generate surface shader */ | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 	if(graph && output->input("Surface")->link) { | 
					
						
							|  |  |  | 		compile_type(shader, shader->graph, SHADER_TYPE_SURFACE); | 
					
						
							|  |  |  | 		og->surface_state.push_back(ss->state()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if(shader->graph_bump) { | 
					
						
							|  |  |  | 			ss->clear_state(); | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 			compile_type(shader, shader->graph_bump, SHADER_TYPE_SURFACE); | 
					
						
							|  |  |  | 			og->surface_state.push_back(ss->state()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 			og->surface_state.push_back(ss->state()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ss->clear_state(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		shader->has_surface = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		og->surface_state.push_back(OSL::ShadingAttribStateRef()); | 
					
						
							|  |  |  | 		og->surface_state.push_back(OSL::ShadingAttribStateRef()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* generate volume shader */ | 
					
						
							|  |  |  | 	if(graph && output->input("Volume")->link) { | 
					
						
							|  |  |  | 		compile_type(shader, shader->graph, SHADER_TYPE_VOLUME); | 
					
						
							|  |  |  | 		shader->has_volume = true; | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 		og->volume_state.push_back(ss->state()); | 
					
						
							|  |  |  | 		og->volume_state.push_back(ss->state()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 		ss->clear_state(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2011-10-12 23:03:12 +00:00
										 |  |  | 		og->volume_state.push_back(OSL::ShadingAttribStateRef()); | 
					
						
							|  |  |  | 		og->volume_state.push_back(OSL::ShadingAttribStateRef()); | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* generate displacement shader */ | 
					
						
							|  |  |  | 	if(graph && output->input("Displacement")->link) { | 
					
						
							|  |  |  | 		compile_type(shader, shader->graph, SHADER_TYPE_DISPLACEMENT); | 
					
						
							|  |  |  | 		shader->has_displacement = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		og->displacement_state.push_back(ss->state()); | 
					
						
							|  |  |  | 		og->displacement_state.push_back(ss->state()); | 
					
						
							|  |  |  | 		ss->clear_state(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		og->displacement_state.push_back(OSL::ShadingAttribStateRef()); | 
					
						
							|  |  |  | 		og->displacement_state.push_back(OSL::ShadingAttribStateRef()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::add(ShaderNode *node, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, float f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_color(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_vector(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 18:08:51 +00:00
										 |  |  | void OSLCompiler::parameter_point(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_normal(const char *name, float3 f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | void OSLCompiler::parameter(const char *name, int f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, const char *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, ustring s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter(const char *name, const Transform& tfm) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 16:31:11 +00:00
										 |  |  | void OSLCompiler::parameter_array(const char *name, const float f[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_color_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_vector_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_normal_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_point_array(const char *name, const float f[][3], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_array(const char *name, const int f[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_array(const char *name, const char * const s[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OSLCompiler::parameter_array(const char *name, const Transform tfm[], int arraylen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-27 11:58:34 +00:00
										 |  |  | #endif /* WITH_OSL */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CCL_NAMESPACE_END | 
					
						
							|  |  |  | 
 |