source: GTP/trunk/App/Demos/Vis/FriendlyCulling/src/shaders/tonemap.cg @ 3149

Revision 3149, 3.5 KB checked in by mattausch, 16 years ago (diff)
Line 
1#include "../shaderenv.h"
2
3
4struct frag
5{
6         // normalized screen position
7        float2 texCoord: TEXCOORD0;
8
9        float2 lt: TEXCOORD1; // left top
10        float2 rb: TEXCOORD2; // right bottom
11        float2 rt: TEXCOORD3; // right top
12        float2 lb: TEXCOORD4; // left bottom
13};
14
15
16struct pixel
17{
18        float4 col: COLOR0;
19};
20
21
22/** Does the first downsampling step and on the same time calculates the
23        intensity.
24*/
25float4 GreyScaleDownSample(frag IN,
26                                                   uniform sampler2D colors,
27                                                   uniform float2 downSampleOffs[4]
28                                                   ): COLOR
29{
30
31        // Compute the average of the 4 necessary samples
32        float average = .0f;
33        float maximum = .0f;
34
35        // the rgb-to-luminance weightings
36        const float3 w = float3(0.299f, 0.587f, 0.114f);
37        //const float3 w = float3(0.2125f, 0.7154f, 0.0721f);
38
39        float4 color;
40
41        for (int i = 0; i < 4; ++ i)
42        {
43                color = tex2D(colors, downSampleOffs[i]);
44                const float intensity = dot(cols[i].rgb, w);
45
46                maximum = max(maximum, intensity);
47                average += log(1e-5f + intensity);
48        }
49
50        average *= 0.25f;
51
52        // Output the luminance to the render target
53        return float4(average, maximum, 0.0f, 1.0f);
54}
55   
56
57/** Used for downsampling the tone map parameters (average loglum, maximum)
58        to the next lower level. This has to be applied until there is only
59        a 1x1 texture which holds the required numbers.
60*/
61float4 DownSampleForToneMapping(frag IN,
62                                                                uniform sampler2D colors,
63                                                                uniform float2 downSampleOffs[4]): COLOR
64{   
65        float average = .0f;
66        float maximum = .0f;
67       
68        float4 color;
69
70        for (int i = 0; i < 4; ++ i)
71        {
72                color = tex2D(colors, downSampleOffs[i]);
73
74                maximum = max(maximum, color.y);
75                average += color.x;
76        }
77
78        average *= 1.0f / (float)NUM_DOWNSAMPLES;
79
80        return float4(average, maximum, 0.0f, 1.0f);
81}
82
83
84
85pixel ToneMap(frag IN,
86                          uniform sampler2D colors,
87                          uniform float imageKey,
88                          uniform float whiteLum,
89                          uniform float middleGrey)
90{
91        pixel OUT;
92       
93        float4 color = tex2D(colors, IN.texCoord);
94
95        const float pixLum = 0.2125f * color.x + 0.7154f * color.y + 0.0721f * color.z;
96       
97        // obtain new image key from highest mipmap level
98        float logLumScaled = tex2Dlod(colors, float4(.5f, .5f, 0, MAX_LOD_LEVEL)).w;
99        float logLum = logLumScaled * LOGLUM_RANGE + MINLOGLUM;
100
101        float newImageKey = max(exp(logLum), 1e-6f);
102
103        // adjust to middle gray
104        const float lum = middleGrey * pixLum / newImageKey;
105        // map to range and calc burnout
106        const float scaleLum = lum * (1.0f + lum / whiteLum * whiteLum) / (1.0f + lum);
107
108        OUT.col = color * scaleLum / pixLum;
109        OUT.col.w = color.w;
110
111        return OUT;
112}
113
114
115pixel CalcAvgLogLum(frag IN, uniform sampler2D colors)
116{
117        ////////////
118        //-- write out logaritmic luminance for tone mapping
119
120        pixel OUT;
121
122        const float4 color = tex2Dlod(colors, float4(IN.texCoord.xy, 0, 0));
123        OUT.col = color;
124
125        // the old loglum is stored in the hightest mipmap-level
126        float oldLogLum = tex2Dlod(colors, float4(IN.texCoord.xy, 0, MAX_LOD_LEVEL)).w;
127
128        // the intensity weights
129        const float3 w = float3(0.299f, 0.587f, 0.114f);
130
131        float lum = dot(color.rgb, w);
132        float logLum = log(max(1e-3f, lum));
133
134        float logLumOffset = MINLOGLUM * INV_LOGLUM_RANGE;
135        float logLumScaled = logLum * INV_LOGLUM_RANGE - logLumOffset;
136
137
138        /////////////
139        //-- exponential smoothing of the tone mapping over time
140
141        const float expFactor = 1e-2f;
142       
143        if (oldLogLum > 1e-3f) // check if loglum from last frame too small (=> tm too bright)
144                OUT.col.w = lerp(oldLogLum, logLumScaled, expFactor);
145        else
146                OUT.col.w = logLumScaled;
147
148        return OUT;
149}
Note: See TracBrowser for help on using the repository browser.