1 /**************************************************************************
2  *
3  * Copyright 2007 VMware, Inc.
4  * All Rights Reserved.
5  *
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:
13  *
14  * The above copyright notice and this permission notice (including the
15  * next paragraph) shall be included in all copies or substantial portions
16  * of the Software.
17  *
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 VMWARE 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.
25  *
26  **************************************************************************/
27 /*
28  * Authors:
29  *   Keith Whitwell <keithw@vmware.com>
30  */
31 
32 #include "util/u_memory.h"
33 #include "i915_context.h"
34 #include "i915_reg.h"
35 #include "i915_state.h"
36 #include "i915_state_inlines.h"
37 
38 /* Convinience function to check immediate state.
39  */
40 
41 static inline void
set_immediate(struct i915_context * i915,unsigned offset,const unsigned state)42 set_immediate(struct i915_context *i915, unsigned offset, const unsigned state)
43 {
44    if (i915->current.immediate[offset] == state)
45       return;
46 
47    i915->current.immediate[offset] = state;
48    i915->immediate_dirty |= 1 << offset;
49    i915->hardware_dirty |= I915_HW_IMMEDIATE;
50 }
51 
52 /***********************************************************************
53  * S0,S1: Vertex buffer state.
54  */
55 static void
upload_S0S1(struct i915_context * i915)56 upload_S0S1(struct i915_context *i915)
57 {
58    unsigned LIS0, LIS1;
59 
60    /* I915_NEW_VBO
61     */
62    LIS0 = i915->vbo_offset;
63 
64    /* Need to force this */
65    if (i915->dirty & I915_NEW_VBO) {
66       i915->immediate_dirty |= 1 << I915_IMMEDIATE_S0;
67       i915->hardware_dirty |= I915_HW_IMMEDIATE;
68    }
69 
70    /* I915_NEW_VERTEX_SIZE
71     */
72    {
73       unsigned vertex_size = i915->current.vertex_info.size;
74 
75       LIS1 = ((vertex_size << 24) | (vertex_size << 16));
76    }
77 
78    set_immediate(i915, I915_IMMEDIATE_S0, LIS0);
79    set_immediate(i915, I915_IMMEDIATE_S1, LIS1);
80 }
81 
82 const struct i915_tracked_state i915_upload_S0S1 = {
83    "imm S0 S1", upload_S0S1, I915_NEW_VBO | I915_NEW_VERTEX_FORMAT};
84 
85 /***********************************************************************
86  * S4: Vertex format, rasterization state
87  */
88 static void
upload_S2S4(struct i915_context * i915)89 upload_S2S4(struct i915_context *i915)
90 {
91    unsigned LIS2, LIS4;
92 
93    /* I915_NEW_VERTEX_FORMAT
94     */
95    {
96       LIS2 = i915->current.vertex_info.hwfmt[1];
97       LIS4 = i915->current.vertex_info.hwfmt[0];
98       assert(LIS4); /* should never be zero? */
99    }
100 
101    LIS4 |= i915->rasterizer->LIS4;
102 
103    set_immediate(i915, I915_IMMEDIATE_S2, LIS2);
104    set_immediate(i915, I915_IMMEDIATE_S4, LIS4);
105 }
106 
107 const struct i915_tracked_state i915_upload_S2S4 = {
108    "imm S2 S4", upload_S2S4, I915_NEW_RASTERIZER | I915_NEW_VERTEX_FORMAT};
109 
110 /***********************************************************************
111  */
112 static void
upload_S5(struct i915_context * i915)113 upload_S5(struct i915_context *i915)
114 {
115    unsigned LIS5 = 0;
116    bool stencil_ccw = i915_stencil_ccw(i915);
117 
118    /* I915_NEW_DEPTH_STENCIL
119     */
120    if (stencil_ccw)
121       LIS5 |= i915->depth_stencil->stencil_LIS5_ccw;
122    else
123       LIS5 |= i915->depth_stencil->stencil_LIS5_cw;
124    /* hope it's safe to set stencil ref value even if stencil test is disabled?
125     */
126    LIS5 |= i915->stencil_ref.ref_value[stencil_ccw] << S5_STENCIL_REF_SHIFT;
127 
128    /* I915_NEW_BLEND
129     */
130    LIS5 |= i915->blend->LIS5;
131 
132 #if 0
133    /* I915_NEW_RASTERIZER
134     */
135    if (i915->rasterizer->LIS7) {
136       LIS5 |= S5_GLOBAL_DEPTH_OFFSET_ENABLE;
137    }
138 #endif
139 
140    set_immediate(i915, I915_IMMEDIATE_S5, LIS5);
141 }
142 
143 const struct i915_tracked_state i915_upload_S5 = {
144    "imm S5", upload_S5,
145    I915_NEW_DEPTH_STENCIL | I915_NEW_BLEND | I915_NEW_RASTERIZER};
146 
147 /***********************************************************************
148  */
149 static void
upload_S6(struct i915_context * i915)150 upload_S6(struct i915_context *i915)
151 {
152    unsigned LIS6 = 0;
153 
154    /* I915_NEW_FRAMEBUFFER
155     */
156    if (i915->framebuffer.cbufs[0])
157       LIS6 |= S6_COLOR_WRITE_ENABLE;
158 
159    /* I915_NEW_BLEND
160     */
161    if (i915->blend) {
162       struct i915_surface *cbuf = i915_surface(i915->framebuffer.cbufs[0]);
163       if (cbuf && cbuf->alpha_in_g)
164          LIS6 |= i915->blend->LIS6_alpha_in_g;
165       else if (cbuf && cbuf->alpha_is_x)
166          LIS6 |= i915->blend->LIS6_alpha_is_x;
167       else
168          LIS6 |= i915->blend->LIS6;
169    }
170 
171    /* I915_NEW_DEPTH
172     */
173    if (i915->depth_stencil)
174       LIS6 |= i915->depth_stencil->depth_LIS6;
175 
176    if (i915->rasterizer)
177       LIS6 |= i915->rasterizer->LIS6;
178 
179    set_immediate(i915, I915_IMMEDIATE_S6, LIS6);
180 }
181 
182 const struct i915_tracked_state i915_upload_S6 = {
183    "imm S6", upload_S6,
184    I915_NEW_BLEND | I915_NEW_DEPTH_STENCIL | I915_NEW_FRAMEBUFFER |
185       I915_NEW_RASTERIZER};
186 
187 /***********************************************************************
188  */
189 static void
upload_S7(struct i915_context * i915)190 upload_S7(struct i915_context *i915)
191 {
192 #if 0
193    unsigned LIS7;
194 
195    /* I915_NEW_RASTERIZER
196     */
197    LIS7 = i915->rasterizer->LIS7;
198 
199    set_immediate(i915, I915_IMMEDIATE_S7, LIS7);
200 #endif
201 }
202 
203 const struct i915_tracked_state i915_upload_S7 = {"imm S7", upload_S7,
204                                                   I915_NEW_RASTERIZER};
205 
206 /***********************************************************************
207  */
208 static const struct i915_tracked_state *atoms[] = {
209    &i915_upload_S0S1, &i915_upload_S2S4, &i915_upload_S5, &i915_upload_S6,
210    &i915_upload_S7};
211 
212 static void
update_immediate(struct i915_context * i915)213 update_immediate(struct i915_context *i915)
214 {
215    int i;
216 
217    for (i = 0; i < ARRAY_SIZE(atoms); i++)
218       if (i915->dirty & atoms[i]->dirty)
219          atoms[i]->update(i915);
220 }
221 
222 struct i915_tracked_state i915_hw_immediate = {
223    "immediate", update_immediate,
224    ~0 /* all state atoms, because we do internal checking */
225 };
226