-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathShader.glsl
258 lines (198 loc) · 6.59 KB
/
Shader.glsl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
//The host must include a #define for MAX_ITRS and TF_RES in this source before
//compiling it. As such, line numbers on the GLSL compiler errors will be off
//by two
#pragma optionNV(fastmath on)
//Because we are reading from 8bit textures, the drivers might want to do math
//at low precision, but we don't want that.
//Convert small if blocks to cmovs
#pragma optionNV(ifcvt all)
#pragma optionNV(inline all)
#pragma optionNV(strict on)
#pragma optionNV(unroll all)
uniform sampler3D scalar_tex;
uniform sampler3D aux_scalar_tex;
uniform sampler3D branch_map_tex;
uniform sampler2D parent_tex;
uniform sampler2D tf_index_tex;
uniform sampler2D depth_tex;
uniform sampler2D saddle_val_tex;
uniform vec4 branch_tex_size;
uniform sampler2D tf_tex;
uniform vec4 tf_tex_size; // { numCols, numRows, 1/numCols, 1/numRows }
uniform vec3 light_vec;
//these are used for rounding. The x component is 1 fewer than the
//number of voxels in that dimension, and the y component is the inverse
//of that
uniform vec2 x_size; // { size-1 , 1/(size-1) }
uniform vec2 y_size;
uniform vec2 z_size;
uniform vec3 x_inch; //small offsets for computing gradient
uniform vec3 y_inch;
uniform vec3 z_inch;
uniform sampler2D global_tf_tex;
// x_down,y_up,etc..
// These round a texture coordinate up or down to the nearest
// texel boundary in a given dimension. Logically:
// x_down( vec3(3.76,2.34,5.55) ) = vec3(3.0,2.34,5.55)
// The problem is that OpenGL puts the texel sample in the center
// of a voxel, really we're rounding to k+0.5 for integers k.
// |---*---|---*---|---*---|---*---|---*---|---*---| real tex space
// |.......|.......|.......|.......|.......| logical tex space
//
vec3 x_down( in vec3 p )
{
p.x *= x_size.r;
p.x = floor(p.x-0.5)+0.5;
p.x *= x_size.g;
return p;
}
vec3 x_up( vec3 p )
{
p.x *= x_size.r;
p.x = ceil(p.x-0.5)+0.5;
p.x *= x_size.g;
return p;
}
vec3 y_down( vec3 p )
{
p.y *= y_size.r;
p.y = floor(p.y-0.5)+0.5;
p.y *= y_size.g;
return p;
}
vec3 y_up( vec3 p )
{
p.y *= y_size.r;
p.y = ceil(p.y-0.5)+0.5;
p.y *= y_size.g;
return p;
}
vec3 z_down( vec3 p )
{
p.z *= z_size.r;
p.z = floor(p.z-0.5)+0.5;
p.z *= z_size.g;
return p;
}
vec3 z_up( vec3 p )
{
p.z *= z_size.r;
p.z = ceil(p.z-0.5)+0.5;
p.z *= z_size.g;
return p;
}
void swap( inout vec3 a, inout vec3 b )
{
vec3 t = a;
a = b;
b = t;
}
vec3 normal()
{
vec3 grad;
vec3 frag_xyz = gl_TexCoord[0].xyz;
grad.x = texture3D( scalar_tex, frag_xyz + x_inch ).a -
texture3D( scalar_tex, frag_xyz - x_inch ).a ;
grad.y = texture3D( scalar_tex, frag_xyz + y_inch ).a -
texture3D( scalar_tex, frag_xyz - y_inch ).a ;
grad.z = texture3D( scalar_tex, frag_xyz + z_inch ).a -
texture3D( scalar_tex, frag_xyz - z_inch ).a ;
return normalize(grad);
}
void main ()
{
float f = texture3D( scalar_tex, gl_TexCoord[0].xyz ).a ;
vec3 frag_xyz = gl_TexCoord[0].xyz;
vec3 lo_pnt = x_down( frag_xyz );
vec3 hi_pnt = x_up ( frag_xyz );
vec3 t_pnt;
if ( texture3D(scalar_tex,lo_pnt).a > texture3D(scalar_tex,hi_pnt).a )
swap(lo_pnt,hi_pnt);
//look for a corner vertex with value no greater than f
t_pnt = y_up (lo_pnt);
lo_pnt = y_down(lo_pnt);
if ( texture3D(scalar_tex,lo_pnt).a > texture3D(scalar_tex,t_pnt).a )
swap(lo_pnt,t_pnt);
t_pnt = z_up (lo_pnt);
lo_pnt = z_down(lo_pnt);
if ( texture3D(scalar_tex,lo_pnt).a > texture3D(scalar_tex,t_pnt).a )
swap(lo_pnt,t_pnt);
//look for a corner vertex with value no less than f
t_pnt = y_up (hi_pnt);
hi_pnt = y_down(hi_pnt);
if ( texture3D(scalar_tex,hi_pnt).a < texture3D(scalar_tex,t_pnt).a )
swap(hi_pnt,t_pnt);
t_pnt = z_up (hi_pnt);
hi_pnt = z_down(hi_pnt);
if ( texture3D(scalar_tex,hi_pnt).a < texture3D(scalar_tex,t_pnt).a )
swap(hi_pnt,t_pnt);
vec2 hi_branch = texture3D(branch_map_tex, hi_pnt).ra;
vec2 lo_branch = texture3D(branch_map_tex, lo_pnt).ra;
float hi_depth = texture2D( depth_tex, hi_branch ).a;
float lo_depth = texture2D( depth_tex, lo_branch ).a;
vec2 the_branch;
#if 1 //this is a good but slow benchmark
for(int i=0; i<MAX_ITRS; ++i) {
float hi_saddle = texture2D( saddle_val_tex, hi_branch ).a;
float lo_saddle = texture2D( saddle_val_tex, lo_branch ).a;
vec2 hi_parent = texture2D( parent_tex, hi_branch ).ra;
vec2 lo_parent = texture2D( parent_tex, lo_branch ).ra;
if (hi_depth > lo_depth) {
if (hi_saddle <= f) {
the_branch = hi_branch;
break;
} else {
hi_branch = hi_parent;
hi_depth -= 1.0/256.0;
}
} else {
if (lo_saddle >= f) {
the_branch = lo_branch;
break;
} else {
lo_branch = lo_parent;
lo_depth -= 1.0/256.0;
}
}
if (all(hi_branch == lo_branch)) {
the_branch = hi_branch;
break;
}
}
#endif
#if 0 //this is an attempt at optimization. Not correct at the moment.
bool go = any(hi_branch != lo_branch);
for(int i=0; go && i<MAX_ITRS; ++i) {
bool hi_depth_greater = hi_depth > lo_depth;
vec2 deeper_branch = hi_depth > lo_depth ? hi_branch : lo_branch;
float deeper_saddle_val =
texture2D( saddle_val_tex, hi_depth>lo_depth?
hi_branch:lo_branch ).a;
vec2 parent_of_deeper =
texture2D( parent_tex, hi_depth>lo_depth?
hi_branch:lo_branch ).ra;
if (hi_depth > lo_depth) {
the_branch = hi_branch;
go = any(hi_branch != lo_branch) && deeper_saddle_val >= f;
hi_branch = parent_of_deeper;
hi_depth -= 1.0/256.0;
} else {
the_branch = lo_branch;
go = any(hi_branch != lo_branch) && deeper_saddle_val <= f;
lo_branch = parent_of_deeper;
lo_depth -= 1.0/256.0;
}
}
#endif
vec2 tf_index = texture2D( tf_index_tex, the_branch ).ra;
tf_index.x += f * float(TF_RES) * tf_tex_size.z ;
if (tf_index.x > 1.0) {
tf_index.x -= 1.0;
tf_index.y += tf_tex_size.w;
}
//the fragment color
vec4 main_color = texture2D( tf_tex, tf_index );
main_color.rgb *= 0.5 + 0.5 * abs(dot(normal(),light_vec));
gl_FragColor = main_color;
gl_FragColor.a *= texture2D( global_tf_tex, vec2(f,0.0) ).a;
}