1 /**************************************************************************
3 * Copyright 2008 Tungsten Graphics, Inc., Cedar Park, Texas.
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the
8 * "Software"), to deal in the Software without restriction, including
9 * without limitation the rights to use, copy, modify, merge, publish,
10 * distribute, sub license, and/or sell copies of the Software, and to
11 * permit persons to whom the Software is furnished to do so, subject to
12 * the following conditions:
14 * The above copyright notice and this permission notice (including the
15 * next paragraph) shall be included in all copies or substantial portions
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21 * IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS BE LIABLE FOR
22 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
26 **************************************************************************/
28 #include "pipe/p_util.h"
29 #include "pipe/p_context.h"
30 #include "draw/draw_context.h"
31 #include "draw/draw_private.h"
32 #include "draw/draw_vbuf.h"
33 #include "draw/draw_vertex.h"
34 #include "draw/draw_pt.h"
37 struct draw_context
*draw
;
39 boolean (*run
)( struct pt_post_vs
*pvs
,
40 struct vertex_header
*vertices
,
48 dot4(const float *a
, const float *b
)
58 static INLINE
unsigned
59 compute_clipmask_gl(const float *clip
, /*const*/ float plane
[][4], unsigned nr
)
64 /* Do the hardwired planes first:
66 if (-clip
[0] + clip
[3] < 0) mask
|= (1<<0);
67 if ( clip
[0] + clip
[3] < 0) mask
|= (1<<1);
68 if (-clip
[1] + clip
[3] < 0) mask
|= (1<<2);
69 if ( clip
[1] + clip
[3] < 0) mask
|= (1<<3);
70 if ( clip
[2] + clip
[3] < 0) mask
|= (1<<4); /* match mesa clipplane numbering - for now */
71 if (-clip
[2] + clip
[3] < 0) mask
|= (1<<5); /* match mesa clipplane numbering - for now */
73 /* Followed by any remaining ones:
75 for (i
= 6; i
< nr
; i
++) {
76 if (dot4(clip
, plane
[i
]) < 0)
84 /* The normal case - cliptest, rhw divide, viewport transform.
86 * Also handle identity viewport here at the expense of a few wasted
89 static boolean
post_vs_cliptest_viewport_gl( struct pt_post_vs
*pvs
,
90 struct vertex_header
*vertices
,
94 struct vertex_header
*out
= vertices
;
95 const float *scale
= pvs
->draw
->viewport
.scale
;
96 const float *trans
= pvs
->draw
->viewport
.translate
;
100 for (j
= 0; j
< count
; j
++) {
101 out
->clip
[0] = out
->data
[0][0];
102 out
->clip
[1] = out
->data
[0][1];
103 out
->clip
[2] = out
->data
[0][2];
104 out
->clip
[3] = out
->data
[0][3];
106 out
->vertex_id
= 0xffff;
108 out
->clipmask
= compute_clipmask_gl(out
->clip
,
110 pvs
->draw
->nr_planes
);
111 clipped
+= out
->clipmask
;
113 if (out
->clipmask
== 0)
116 float w
= 1.0f
/ out
->data
[0][3];
118 /* Viewport mapping */
119 out
->data
[0][0] = out
->data
[0][0] * w
* scale
[0] + trans
[0];
120 out
->data
[0][1] = out
->data
[0][1] * w
* scale
[1] + trans
[1];
121 out
->data
[0][2] = out
->data
[0][2] * w
* scale
[2] + trans
[2];
125 out
= (struct vertex_header
*)( (char *)out
+ stride
);
133 /* If bypass_clipping is set, skip cliptest and rhw divide.
135 static boolean
post_vs_viewport( struct pt_post_vs
*pvs
,
136 struct vertex_header
*vertices
,
140 struct vertex_header
*out
= vertices
;
141 const float *scale
= pvs
->draw
->viewport
.scale
;
142 const float *trans
= pvs
->draw
->viewport
.translate
;
145 debug_printf("%s\n", __FUNCTION__
);
146 for (j
= 0; j
< count
; j
++) {
147 /* Viewport mapping only, no cliptest/rhw divide
149 out
->data
[0][0] = out
->data
[0][0] * scale
[0] + trans
[0];
150 out
->data
[0][1] = out
->data
[0][1] * scale
[1] + trans
[1];
151 out
->data
[0][2] = out
->data
[0][2] * scale
[2] + trans
[2];
153 out
= (struct vertex_header
*)((char *)out
+ stride
);
160 /* If bypass_clipping is set and we have an identity viewport, nothing
163 static boolean
post_vs_none( struct pt_post_vs
*pvs
,
164 struct vertex_header
*vertices
,
168 debug_printf("%s\n", __FUNCTION__
);
172 boolean
draw_pt_post_vs_run( struct pt_post_vs
*pvs
,
173 struct vertex_header
*pipeline_verts
,
177 return pvs
->run( pvs
, pipeline_verts
, count
, stride
);
181 void draw_pt_post_vs_prepare( struct pt_post_vs
*pvs
,
182 boolean bypass_clipping
,
183 boolean identity_viewport
,
186 if (bypass_clipping
) {
187 if (identity_viewport
)
188 pvs
->run
= post_vs_none
;
190 pvs
->run
= post_vs_viewport
;
194 pvs
->run
= post_vs_cliptest_viewport_gl
;
199 struct pt_post_vs
*draw_pt_post_vs_create( struct draw_context
*draw
)
201 struct pt_post_vs
*pvs
= CALLOC_STRUCT( pt_post_vs
);
210 void draw_pt_post_vs_destroy( struct pt_post_vs
*pvs
)