/* * Copyright 2011-2013 Blender Foundation * * 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 * * http://www.apache.org/licenses/LICENSE-2.0 * * 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 * limitations under the License. */ #ifndef __GRAPH_H__ #define __GRAPH_H__ #include "node.h" #include "kernel_types.h" #include "util_list.h" #include "util_map.h" #include "util_param.h" #include "util_set.h" #include "util_types.h" #include "util_vector.h" CCL_NAMESPACE_BEGIN class AttributeRequestSet; class Scene; class Shader; class ShaderInput; class ShaderOutput; class ShaderNode; class ShaderGraph; class SVMCompiler; class OSLCompiler; class OutputNode; /* Bump * * For bump mapping, a node may be evaluated multiple times, using different * samples to reconstruct the normal, this indicates the sample position */ enum ShaderBump { SHADER_BUMP_NONE, SHADER_BUMP_CENTER, SHADER_BUMP_DX, SHADER_BUMP_DY }; /* Identifiers for some special node types. * * The graph needs to identify these in the clean function. * Cannot use dynamic_cast, as this is disabled for OSL. */ enum ShaderNodeSpecialType { SHADER_SPECIAL_TYPE_NONE, SHADER_SPECIAL_TYPE_PROXY, SHADER_SPECIAL_TYPE_AUTOCONVERT, SHADER_SPECIAL_TYPE_GEOMETRY, SHADER_SPECIAL_TYPE_SCRIPT, SHADER_SPECIAL_TYPE_IMAGE_SLOT, SHADER_SPECIAL_TYPE_CLOSURE, SHADER_SPECIAL_TYPE_COMBINE_CLOSURE, SHADER_SPECIAL_TYPE_OUTPUT, SHADER_SPECIAL_TYPE_BUMP, }; /* Input * * Input socket for a shader node. May be linked to an output or not. If not * linked, it will either get a fixed default value, or e.g. a texture * coordinate. */ class ShaderInput { public: ShaderInput(ShaderNode *parent, const char *name, SocketType::Type type); ustring name() { return name_; } int flags() { return flags_; } SocketType::Type type() { return type_; } float3& value() { return value_; } float& value_float() { return value_.x; } ustring& value_string() { return value_string_; } ustring name_; SocketType::Type type_; ShaderNode *parent; ShaderOutput *link; float3 value_; ustring value_string_; int stack_offset; /* for SVM compiler */ int flags_; }; /* Output * * Output socket for a shader node. */ class ShaderOutput { public: ShaderOutput(ShaderNode *parent, const char *name, SocketType::Type type); ustring name() { return name_; } SocketType::Type type() { return type_; } ustring name_; SocketType::Type type_; ShaderNode *parent; vector links; int stack_offset; /* for SVM compiler */ }; /* Node * * Shader node in graph, with input and output sockets. This is the virtual * base class for all node types. */ class ShaderNode { public: explicit ShaderNode(const char *name); virtual ~ShaderNode(); ShaderInput *input(const char *name); ShaderOutput *output(const char *name); ShaderInput *input(ustring name); ShaderOutput *output(ustring name); ShaderInput *add_input(const char *name, SocketType::Type type, float value=0.0f, int flags=0); ShaderInput *add_input(const char *name, SocketType::Type type, float3 value, int flags=0); ShaderOutput *add_output(const char *name, SocketType::Type type); virtual ShaderNode *clone() const = 0; virtual void attributes(Shader *shader, AttributeRequestSet *attributes); virtual void compile(SVMCompiler& compiler) = 0; virtual void compile(OSLCompiler& compiler) = 0; /* ** Node optimization ** */ /* Check whether the node can be replaced with single constant. */ virtual bool constant_fold(ShaderGraph * /*graph*/, ShaderOutput * /*socket*/, float3 * /*optimized_value*/) { return false; } /* Simplify settings used by artists to the ones which are simpler to * evaluate in the kernel but keep the final result unchanged. */ virtual void simplify_settings(Scene * /*scene*/) {}; virtual bool has_surface_emission() { return false; } virtual bool has_surface_transparent() { return false; } virtual bool has_surface_bssrdf() { return false; } virtual bool has_bssrdf_bump() { return false; } virtual bool has_spatial_varying() { return false; } virtual bool has_object_dependency() { return false; } virtual bool has_integrator_dependency() { return false; } vector inputs; vector outputs; ustring name; /* name, not required to be unique */ int id; /* index in graph node array */ ShaderBump bump; /* for bump mapping utility */ ShaderNodeSpecialType special_type; /* special node type */ /* ** Selective nodes compilation ** */ /* TODO(sergey): More explicitly mention in the function names * that those functions are for selective compilation only? */ /* Nodes are split into several groups, group of level 0 contains * nodes which are most commonly used, further levels are extension * of previous one and includes less commonly used nodes. */ virtual int get_group() { return NODE_GROUP_LEVEL_0; } /* Node feature are used to disable huge nodes inside the group, * so it's possible to disable huge nodes inside of the required * nodes group. */ virtual int get_feature() { return bump == SHADER_BUMP_NONE ? 0 : NODE_FEATURE_BUMP; } /* Get closure ID to which the node compiles into. */ virtual ClosureType get_closure_type() { return CLOSURE_NONE_ID; } /* Check whether settings of the node equals to another one. * * This is mainly used to check whether two nodes can be merged * together. Meaning, runtime stuff like node id and unbound slots * will be ignored for comparison. * * NOTE: If some node can't be de-duplicated for whatever reason it * is to be handled in the subclass. */ virtual bool equals(const ShaderNode *other) { return name == other->name && bump == other->bump; } }; /* Node definition utility macros */ #define SHADER_NODE_CLASS(type) \ type(); \ virtual ShaderNode *clone() const { return new type(*this); } \ virtual void compile(SVMCompiler& compiler); \ virtual void compile(OSLCompiler& compiler); \ #define SHADER_NODE_NO_CLONE_CLASS(type) \ type(); \ virtual void compile(SVMCompiler& compiler); \ virtual void compile(OSLCompiler& compiler); \ #define SHADER_NODE_BASE_CLASS(type) \ virtual ShaderNode *clone() const { return new type(*this); } \ virtual void compile(SVMCompiler& compiler); \ virtual void compile(OSLCompiler& compiler); \ class ShaderNodeIDComparator { public: bool operator()(const ShaderNode *n1, const ShaderNode *n2) const { return n1->id < n2->id; } }; typedef set ShaderNodeSet; typedef map ShaderNodeMap; /* Graph * * Shader graph of nodes. Also does graph manipulations for default inputs, * bump mapping from displacement, and possibly other things in the future. */ class ShaderGraph { public: list nodes; size_t num_node_ids; bool finalized; ShaderGraph(); ~ShaderGraph(); ShaderGraph *copy(); ShaderNode *add(ShaderNode *node); OutputNode *output(); void connect(ShaderOutput *from, ShaderInput *to); void disconnect(ShaderInput *to); void relink(ShaderNode *node, ShaderOutput *from, ShaderOutput *to); void remove_proxy_nodes(); void finalize(Scene *scene, bool do_bump = false, bool do_osl = false, bool do_simplify = false); int get_num_closures(); void dump_graph(const char *filename); protected: typedef pair NodePair; void find_dependencies(ShaderNodeSet& dependencies, ShaderInput *input); void clear_nodes(); void copy_nodes(ShaderNodeSet& nodes, ShaderNodeMap& nnodemap); void break_cycles(ShaderNode *node, vector& visited, vector& on_stack); void bump_from_displacement(); void refine_bump_nodes(); void default_inputs(bool do_osl); void transform_multi_closure(ShaderNode *node, ShaderOutput *weight_out, bool volume); /* Graph simplification routines. */ void clean(Scene *scene); void constant_fold(); void simplify_settings(Scene *scene); void deduplicate_nodes(); }; CCL_NAMESPACE_END #endif /* __GRAPH_H__ */