Cycles: svn merge -r41182:41205 ^/trunk/blender
[blender.git] / intern / cycles / kernel / svm / svm_fresnel.h
1 /*
2  * Copyright 2011, Blender Foundation.
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  */
18
19 CCL_NAMESPACE_BEGIN
20
21 /* Fresnel Node */
22
23 __device void svm_node_fresnel(ShaderData *sd, float *stack, uint ior_offset, uint ior_value, uint out_offset)
24 {
25         float eta = (stack_valid(ior_offset))? stack_load_float(stack, ior_offset): __int_as_float(ior_value);
26         eta = fmaxf(eta, 1.0f + 1e-5f);
27         eta = (sd->flag & SD_BACKFACING)? 1.0f/eta: eta;
28
29         float f = fresnel_dielectric_cos(dot(sd->I, sd->N), eta);
30
31         stack_store_float(stack, out_offset, f);
32 }
33
34 /* Blend Weight Node */
35
36 __device void svm_node_blend_weight(ShaderData *sd, float *stack, uint4 node)
37 {
38         uint blend_offset = node.y;
39         uint blend_value = node.z;
40         float blend = (stack_valid(blend_offset))? stack_load_float(stack, blend_offset): __int_as_float(blend_value);
41
42         uint type, out_offset;
43         decode_node_uchar4(node.w, &type, &out_offset, NULL, NULL);
44
45         float f;
46
47         if(type == NODE_BLEND_WEIGHT_FRESNEL) {
48                 float eta = fmaxf(1.0f - blend, 1e-5f);
49                 eta = (sd->flag & SD_BACKFACING)? eta: 1.0f/eta;
50
51                 f = fresnel_dielectric_cos(dot(sd->I, sd->N), eta);
52         }
53         else {
54                 f = fabsf(dot(sd->I, sd->N));
55
56                 if(blend != 0.5f) {
57                         blend = clamp(blend, 0.0f, 1.0f);
58                         blend = (blend < 0.5f)? 2.0f*blend: 0.5f/(1.0f - blend);
59
60                         f = powf(f, blend);
61                 }
62
63                 f = 1.0f - f;
64         }
65
66         stack_store_float(stack, out_offset, f);
67 }
68
69 CCL_NAMESPACE_END
70